library(tidyverse)
Registered S3 methods overwritten by 'dbplyr':
  method         from
  print.tbl_lazy     
  print.tbl_sql      
── Attaching packages ──────────────────────── tidyverse 1.3.1 ──
✔ ggplot2 3.3.6     ✔ purrr   0.3.4
✔ tibble  3.1.7     ✔ dplyr   1.0.9
✔ tidyr   1.2.0     ✔ stringr 1.4.0
✔ readr   2.1.2     ✔ forcats 0.5.1
── Conflicts ─────────────────────────── tidyverse_conflicts() ──
✖ dplyr::filter() masks stats::filter()
✖ dplyr::lag()    masks stats::lag()
library(scales)

Attaching package: ‘scales’

The following object is masked from ‘package:purrr’:

    discard

The following object is masked from ‘package:readr’:

    col_factor
library(plotly)
Registered S3 method overwritten by 'data.table':
  method           from
  print.data.table     
Registered S3 method overwritten by 'htmlwidgets':
  method           from         
  print.htmlwidget tools:rstudio

Attaching package: ‘plotly’

The following object is masked from ‘package:ggplot2’:

    last_plot

The following object is masked from ‘package:stats’:

    filter

The following object is masked from ‘package:graphics’:

    layout
library(lubridate)

Attaching package: ‘lubridate’

The following objects are masked from ‘package:base’:

    date, intersect, setdiff, union

Beds

beds <- read_csv("raw_data/non_covid_raw_data/beds_by_nhs_board_of_treatment_and_specialty.csv") %>% janitor::clean_names()
Warning: One or more parsing issues, see `problems()` for detailsRows: 30458 Columns: 20── Column specification ─────────────────────────────────────────
Delimiter: ","
chr (10): Quarter, QuarterQF, HB, HBQF, Location, LocationQF,...
dbl  (5): AllStaffedBeddays, TotalOccupiedBeddays, AverageAva...
lgl  (5): AllStaffedBeddaysQF, TotalOccupiedBeddaysQF, Averag...
ℹ Use `spec()` to retrieve the full column specification for this data.
ℹ Specify the column types or set `show_col_types = FALSE` to quiet this message.
beds %>% 
  filter(specialty_name == "All Acute") %>% 
  ggplot(aes(x = quarter, y = percentage_occupancy))+
  geom_line(aes(colour = hb), group = 1)+
  facet_wrap(~ hb)

beds <- beds %>% 
  select(-c(2,4,6,8,10,12,14,16,18,20)) %>% 
  filter(!hb %in% c("SB0801", "S92000003"))

beds <- beds %>% 
  filter(!hb %in% c("SB0801", "S92000003")) %>% 
  filter(hb == location)
  
# beds %>% 
# count(specialty_name)

a_e_beds <- beds %>% 
  filter(specialty_name == "Accident & Emergency")
# a&e percentage occupancy by hb over time
a_e_beds %>% 
  group_by(quarter, hb) %>% 
  summarise(mean_perc_occ = mean(percentage_occupancy, na.rm = TRUE)) %>% 
  ggplot(aes(x = quarter, y = mean_perc_occ))+
  geom_line(aes(group = hb, colour = hb))+
  geom_point()+
  theme(axis.text.x = element_text(angle = 45))
`summarise()` has grouped output by 'quarter'. You can override using the `.groups` argument.

a_e_beds %>% 
  count(hb)
#all time bed occupancy percentage for health boards
a_e_beds %>% 
  group_by(quarter, hb) %>% 
  summarise(mean_perc_occ = mean(percentage_occupancy, na.rm = TRUE)) %>% 
ungroup() %>% 
  group_by(hb) %>% 
  summarise(avg_per_occ_all_time = mean(mean_perc_occ)) %>% 
  arrange(desc(avg_per_occ_all_time)) %>% 
  ggplot(aes(x = hb, y = avg_per_occ_all_time))+
  geom_col()+
  geom_point()+
  theme(axis.text.x = element_text(angle = 45))
`summarise()` has grouped output by 'quarter'. You can override using the `.groups` argument.

# workout the ten largest 
ten_largest_specialities <- beds %>%
  group_by(specialty_name) %>% 
  summarise(mean_avail_staffed_beds = mean(average_available_staffed_beds)) %>% 
  arrange(desc(mean_avail_staffed_beds)) %>% 
  slice_max(mean_avail_staffed_beds, n=10) %>% 
  select(1) %>% 
  pull()



# bed percentage availablity for top ten largest specialities
beds %>%
  filter(specialty_name %in% ten_largest_specialities) %>% 
  group_by(quarter, specialty_name) %>%
  summarise(mean_perc_occ = mean(percentage_occupancy)) %>% 
  ggplot(aes(x = quarter, y = mean_perc_occ))+
  geom_line(aes(colour = specialty_name, group = specialty_name))+
  geom_point()+
  theme(axis.text.x = element_text(angle = 45, hjust = 1))
`summarise()` has grouped output by 'quarter'. You can override using the `.groups` argument.

  
# bed percentage availablity for all acute
  beds %>%
  filter(specialty_name == "All Acute") %>% 
  group_by(quarter, specialty_name) %>%
  summarise(mean_perc_occ = mean(percentage_occupancy)) %>% 
  ggplot(aes(x = quarter, y = mean_perc_occ))+
  geom_line(aes(colour = specialty_name, group = specialty_name))+
  geom_point()+
  theme(axis.text.x = element_text(angle = 45, hjust = 1))
`summarise()` has grouped output by 'quarter'. You can override using the `.groups` argument.

  
  # bed percentage availability for intensive care
beds %>%
  filter(specialty_name == "Intensive Care Medicine") %>% 
  group_by(quarter, specialty_name) %>%
  summarise(mean_perc_occ = mean(percentage_occupancy)) %>% 
  ggplot(aes(x = quarter, y = mean_perc_occ))+
  geom_line(aes(colour = specialty_name, group = specialty_name))+
  geom_point()+
  theme(axis.text.x = element_text(angle = 45, hjust = 1))
`summarise()` has grouped output by 'quarter'. You can override using the `.groups` argument.

NA
NA
NA
a_e_beds %>% 
  group_by(quarter) %>% 
  summarise(mean_available_beds = mean(average_available_staffed_beds, na.rm = TRUE)) %>% 
  ggplot(aes(x = quarter, y = mean_available_beds))+
  geom_line(group ="1")+
  geom_point()+
  theme(axis.text.x = element_text(angle = 45))


age sex

age_sex <- read_csv("raw_data/non_covid_raw_data/inpatient_and_daycase_by_nhs_board_of_treatment_age_and_sex.csv") %>% janitor::clean_names()
Rows: 129393 Columns: 18── Column specification ─────────────────────────────────────────
Delimiter: ","
chr (12): Quarter, QuarterQF, HB, HBQF, Location, LocationQF,...
dbl  (6): Episodes, LengthOfEpisode, AverageLengthOfEpisode, ...
ℹ Use `spec()` to retrieve the full column specification for this data.
ℹ Specify the column types or set `show_col_types = FALSE` to quiet this message.
health_board_names <- read_csv("raw_data/non_covid_raw_data/health_board_names.csv")
Rows: 18 Columns: 5── Column specification ─────────────────────────────────────────
Delimiter: ","
chr (3): HB, HBName, Country
dbl (2): HBDateEnacted, HBDateArchived
ℹ Use `spec()` to retrieve the full column specification for this data.
ℹ Specify the column types or set `show_col_types = FALSE` to quiet this message.
age_sex %>% 
  count(hb)

season_age_sex <- age_sex %>% 
  mutate(date = yq(quarter),
         year = year(date),
         month = month(date, label = TRUE, abbr = FALSE),
         season = case_when(
           str_detect(month, "December") ~ "Winter",
           str_detect(month, "January") ~ "Winter",
           str_detect(month, "February") ~ "Winter",
           str_detect(month, "March") ~ "Spring",
           str_detect(month, "April") ~ "Spring",
           str_detect(month, "May") ~ "Spring",
           str_detect(month, "June") ~ "Summer",
           str_detect(month, "July") ~ "Summer",
           str_detect(month, "August") ~ "Summer",
           str_detect(month, "September") ~ "Autumn",
           str_detect(month, "October") ~ "Autumn",
           str_detect(month, "November") ~ "Autumn"),
         season = factor(season, order = TRUE)) 

library(lubridate)
library(zoo)

Attaching package: ‘zoo’

The following objects are masked from ‘package:base’:

    as.Date, as.Date.numeric
# change quarter column into the date at the start of each quarter
 age_sex <-  age_sex %>% 
    mutate(quarter = yq(quarter))

 # shows the total length of stay by age bracket for emergency inpatients
age_sex %>% 
  filter(admission_type == "Emergency Inpatients") %>% 
  group_by(quarter, age) %>% 
  summarise(total_length_of_stay = sum(length_of_stay)) %>% 
  ggplot(aes(x = quarter, y = total_length_of_stay))+
  geom_line(aes(colour = age))+
  theme(axis.text.x = element_text(angle = 45))
`summarise()` has grouped output by 'quarter'. You can override using the `.groups` argument.

age_sex %>% 
count(admission_type)

age_sex
 # shows the total length of stay by age bracket for elective inpatients
age_sex %>% 
  filter(admission_type == "Elective Inpatients") %>% 
  group_by(quarter, age) %>% 
  summarise(total_length_of_stay = sum(length_of_stay)) %>% 
  ggplot(aes(x = quarter, y = total_length_of_stay))+
  geom_line(aes(colour = age))+
  theme(axis.text.x = element_text(angle = 45))
`summarise()` has grouped output by 'quarter'. You can override using the `.groups` argument.

 # shows the mean length of stay by age bracket for emergency inpatients
#can facet by sex also if required
age_sex %>% 
  filter(admission_type == "Emergency Inpatients") %>% 
  group_by(quarter, age) %>% 
  summarise(avg_length_of_stay = mean(average_length_of_stay, na.rm = TRUE)) %>% 
  ggplot(aes(x = quarter, y = avg_length_of_stay))+
  geom_line(aes(colour = age, group = age))+
  theme(axis.text.x = element_text(angle = 45))
`summarise()` has grouped output by 'quarter'. You can override using the `.groups` argument.

  #facet_wrap( ~ sex)

 # shows the mean length of stay by age bracket for elective inpatients
# can facet by sex 
age_sex %>% 
  filter(admission_type == "Elective Inpatients") %>% 
  group_by(quarter, age) %>% 
  summarise(avg_length_of_stay = mean(average_length_of_stay, na.rm = TRUE)) %>% 
  ggplot(aes(x = quarter, y = avg_length_of_stay))+
  geom_line(aes(colour = age, group = age))+
  theme(axis.text.x = element_text(angle = 45))
`summarise()` has grouped output by 'quarter'. You can override using the `.groups` argument.

#   #facet_wrap(~ sex)
age_sex %>% 
  mutate(date = yq(quarter),
         year = year(date),
         month = month(date, label = TRUE, abbr = FALSE),
         season = case_when(
           str_detect(month, "December") ~ "Winter",
           str_detect(month, "January") ~ "Winter",
           str_detect(month, "February") ~ "Winter",
           str_detect(month, "March") ~ "Spring",
           str_detect(month, "April") ~ "Spring",
           str_detect(month, "May") ~ "Spring",
           str_detect(month, "June") ~ "Summer",
           str_detect(month, "July") ~ "Summer",
           str_detect(month, "August") ~ "Summer",
           str_detect(month, "September") ~ "Autumn",
           str_detect(month, "October") ~ "Autumn",
           str_detect(month, "November") ~ "Autumn"),
         season = factor(season, order = TRUE)) 
Warning: All formats failed to parse. No formats found.

age_sex %>% 
  filter(admission_type == "Emergency Inpatients") %>% 
  group_by(quarter, sex) %>% 
  summarise(avg_length_of_stay = mean(average_length_of_stay, na.rm = TRUE)) %>% 
  ggplot(aes(x = quarter, y = avg_length_of_stay))+
  geom_line(aes(colour = sex, group = sex))+
  theme(axis.text.x = element_text(angle = 45))+
  geom_smooth(aes(colour = sex, group = sex), se = FALSE)+
  geom_point(aes(colour = sex), size = 0.5)+
  labs(title = "Emergency Inpatient by gender and average length of stay")
`summarise()` has grouped output by 'quarter'. You can override using the `.groups` argument.

age_sex %>% 
  filter(admission_type == "Elective Inpatients") %>% 
  group_by(quarter, sex) %>% 
  summarise(avg_length_of_stay = mean(average_length_of_stay, na.rm = TRUE)) %>% 
  ggplot(aes(x = quarter, y = avg_length_of_stay))+
  geom_line(aes(colour = sex, group = sex))+
  theme(axis.text.x = element_text(angle = 45))+
  geom_smooth(aes(colour = sex, group = sex), se = FALSE)+
  geom_point(aes(colour = sex), size = 0.5)+
  labs(title = "Elective Inpatient by gender and average length of stay")
`summarise()` has grouped output by 'quarter'. You can override using the `.groups` argument.

# emergency inpatient by age and avg length of stay
age_sex %>% 
  filter(admission_type == "Emergency Inpatients") %>% 
  group_by(quarter, age) %>% 
  summarise(avg_length_of_stay = mean(average_length_of_stay, na.rm = TRUE)) %>% 
  ggplot(aes(x = quarter, y = avg_length_of_stay))+
  geom_line(aes(colour = age, group = age))+
  theme(axis.text.x = element_text(angle = 45))+
  #geom_smooth(aes(colour = sex, group = sex), se = FALSE)+
  geom_point(aes(colour = age), size = 0.5)+
  labs(title = "Emergency inpatient by age and avg length of stay")
`summarise()` has grouped output by 'quarter'. You can override using the `.groups` argument.

# elective inpatient by age and avg length of stay
age_sex %>% 
  filter(admission_type == "Elective Inpatients") %>% 
  group_by(quarter, age) %>% 
  summarise(avg_length_of_stay = mean(average_length_of_stay, na.rm = TRUE)) %>% 
  ggplot(aes(x = quarter, y = avg_length_of_stay))+
  geom_line(aes(colour = age, group = age))+
  theme(axis.text.x = element_text(angle = 45))+
  #geom_smooth(aes(colour = sex, group = sex), se = FALSE)+
  geom_point(aes(colour = age), size = 0.5)+
  labs(title = "Elective inpatient by age and avg length of stay")
`summarise()` has grouped output by 'quarter'. You can override using the `.groups` argument.

# emergency inpatient by age and avg length of epsiode
age_sex %>% 
  filter(admission_type == "Emergency Inpatients") %>% 
  group_by(quarter, age) %>% 
  summarise(avg_length_of_episode = mean(average_length_of_episode, na.rm = TRUE)) %>% 
  ggplot(aes(x = quarter, y = avg_length_of_episode))+
  geom_line(aes(colour = age, group = age))+
  theme(axis.text.x = element_text(angle = 45))+
  #geom_smooth(aes(colour = sex, group = sex), se = FALSE)+
  geom_point(aes(colour = age), size = 0.5)+
  labs(title = "Emergency inpatient by age and avg length of episode")
`summarise()` has grouped output by 'quarter'. You can override using the `.groups` argument.

# elective inpatient by age and avg length of episode
age_sex %>% 
  filter(admission_type == "Elective Inpatients") %>% 
  group_by(quarter, age) %>% 
  summarise(avg_length_of_episode = mean(average_length_of_episode, na.rm = TRUE)) %>% 
  ggplot(aes(x = quarter, y = avg_length_of_episode))+
  geom_line(aes(colour = age, group = age))+
  theme(axis.text.x = element_text(angle = 45))+
  #geom_smooth(aes(colour = sex, group = sex), se = FALSE)+
  geom_point(aes(colour = age), size = 0.5)+
  labs(title = "Elective inpatient by age and avg length of episode")
`summarise()` has grouped output by 'quarter'. You can override using the `.groups` argument.

# emergency inpatient by age and avg episodes
age_sex %>% 
  filter(admission_type == "Emergency Inpatients") %>% 
  group_by(quarter, age) %>% 
  summarise(avg_episodes = mean(episodes, na.rm = TRUE)) %>% 
  ggplot(aes(x = quarter, y = avg_episodes))+
  geom_line(aes(colour = age, group = age))+
  theme(axis.text.x = element_text(angle = 45))+
  #geom_smooth(aes(colour = sex, group = sex), se = FALSE)+
  geom_point(aes(colour = age), size = 0.5)+
  labs(title = "Emergency inpatient by age and avg episodes")
`summarise()` has grouped output by 'quarter'. You can override using the `.groups` argument.

# emergency inpatient by age and avg episodes
age_sex %>% 
  filter(admission_type == "Elective Inpatients") %>% 
  group_by(quarter, age) %>% 
  summarise(avg_episodes = mean(episodes, na.rm = TRUE)) %>% 
  ggplot(aes(x = quarter, y = avg_episodes))+
  geom_line(aes(colour = age, group = age))+
  theme(axis.text.x = element_text(angle = 45))+
  #geom_smooth(aes(colour = sex, group = sex), se = FALSE)+
  geom_point(aes(colour = age), size = 0.5)+
  labs(title = "Elective inpatient by age and avg episodes")
`summarise()` has grouped output by 'quarter'. You can override using the `.groups` argument.

# Plot comparison of Emergency vs Elective submissions
age_sex %>% 
  filter(admission_type %in% c("Emergency Inpatients", "Elective Inpatients")) %>% 
  group_by(quarter, admission_type) %>% 
  summarise(avg_stay = mean(average_length_of_stay, na.rm = TRUE)) %>% 
  ggplot(aes(x = quarter, y = avg_stay, colour = admission_type))+
  geom_line(aes(group = admission_type))+
  geom_point()+
  theme(axis.text.x = element_text(angle = 45, hjust = 1))
`summarise()` has grouped output by 'quarter'. You can override using the `.groups` argument.

# avg_stay by admission type
age_sex %>% 
  group_by(quarter, admission_type) %>% 
  summarise(avg_stay = mean(average_length_of_stay, na.rm = TRUE)) %>% 
  ggplot(aes(x = quarter, y = avg_stay, colour = admission_type))+
  geom_line(aes(group = admission_type))+
  geom_point()+
  theme(axis.text.x = element_text(angle = 45, hjust = 1))
`summarise()` has grouped output by 'quarter'. You can override using the `.groups` argument.

# avg_stay for all types
age_sex %>% 
  group_by(quarter) %>% 
  summarise(avg_stay = mean(average_length_of_stay, na.rm = TRUE)) %>% 
  ggplot(aes(x = quarter, y = avg_stay, group = 1))+
  geom_line(aes(group = 1))+
  geom_point()+
  theme(axis.text.x = element_text(angle = 45, hjust = 1))


# number of stays for all types
age_sex %>% 
  group_by(quarter) %>% 
  summarise(total_stays = sum(stays, na.rm = TRUE)) %>% 
  ggplot(aes(x = quarter, y = total_stays, group = 1))+
  geom_line(aes(group = 1))+
  geom_point()+
  theme(axis.text.x = element_text(angle = 45, hjust = 1))

simd <- read_csv("raw_data/non_covid_raw_data/inpatient_and_daycase_by_nhs_board_of_treatment_and_simd.csv") %>% janitor::clean_names()
Rows: 40821 Columns: 18── Column specification ─────────────────────────────────────────
Delimiter: ","
chr (11): Quarter, QuarterQF, HB, HBQF, Location, LocationQF,...
dbl  (7): SIMD, Episodes, LengthOfEpisode, AverageLengthOfEpi...
ℹ Use `spec()` to retrieve the full column specification for this data.
ℹ Specify the column types or set `show_col_types = FALSE` to quiet this message.
#total episodes(hospitalisations?) by simd value
simd %>% 
  drop_na(simd) %>%
  mutate(simd = as.factor(simd)) %>% # gives each simd a separate colour
  group_by(quarter, simd) %>% 
  summarise(total_episodes = sum(episodes, na.rm = TRUE)) %>% 
  ggplot(aes(x = quarter, y = total_episodes, group = simd))+
  geom_line(aes(colour = simd))+
  theme(axis.text.x = element_text(angle = 45, hjust = 1))+
  scale_y_continuous(labels = scales::comma)
`summarise()` has grouped output by 'quarter'. You can override using the `.groups` argument.

  
#avg number of episodes(hospitalisations?) by simd value
simd %>% 
  drop_na(simd) %>%
  mutate(simd = as.factor(simd)) %>% # gives each simd a separate colour
  group_by(quarter, simd) %>% 
  summarise(avg_episodes = mean(episodes, na.rm = TRUE)) %>% 
  ggplot(aes(x = quarter, y = avg_episodes, group = simd))+
  geom_line(aes(colour = simd))+
  theme(axis.text.x = element_text(angle = 45, hjust = 1))+
  scale_y_continuous(labels = scales::comma)
`summarise()` has grouped output by 'quarter'. You can override using the `.groups` argument.

#avg stays by simd value
simd %>% 
  drop_na(simd) %>%
  mutate(simd = as.factor(simd)) %>% # gives each simd a separate colour
  group_by(quarter, simd) %>% 
  summarise(avg_stays = mean(stays, na.rm = TRUE)) %>% 
  ggplot(aes(x = quarter, y = avg_stays, group = simd))+
  geom_line(aes(colour = simd))+
  theme(axis.text.x = element_text(angle = 45, hjust = 1))+
  scale_y_continuous(labels = scales::comma)
`summarise()` has grouped output by 'quarter'. You can override using the `.groups` argument.

NA
#total episodes(hospitalisations?) by simd value
simd %>% 
  drop_na(simd) %>%
  filter(admission_type == "Emergency Inpatients") %>% 
  mutate(simd = as.factor(simd)) %>% # gives each simd a separate colour
  group_by(quarter, simd) %>% 
  summarise(total_episodes = sum(episodes, na.rm = TRUE)) %>% 
  ggplot(aes(x = quarter, y = total_episodes, group = simd))+
  geom_line(aes(colour = simd))+
  theme(axis.text.x = element_text(angle = 45, hjust = 1))+
  scale_y_continuous(labels = scales::comma)
`summarise()` has grouped output by 'quarter'. You can override using the `.groups` argument.

  
#avg number of episodes(hospitalisations?) by simd value
simd %>% 
  drop_na(simd) %>%
  filter(admission_type == "Emergency Inpatients") %>% 
  mutate(simd = as.factor(simd)) %>% # gives each simd a separate colour
  group_by(quarter, simd) %>% 
  summarise(avg_episodes = mean(episodes, na.rm = TRUE)) %>% 
  ggplot(aes(x = quarter, y = avg_episodes, group = simd))+
  geom_line(aes(colour = simd))+
  theme(axis.text.x = element_text(angle = 45, hjust = 1))+
  scale_y_continuous(labels = scales::comma)
`summarise()` has grouped output by 'quarter'. You can override using the `.groups` argument.

#avg stays by simd value
simd %>% 
  drop_na(simd) %>%
  filter(admission_type == "Emergency Inpatients") %>% 
  mutate(simd = as.factor(simd)) %>% # gives each simd a separate colour
  group_by(quarter, simd) %>% 
  summarise(avg_stays = mean(stays, na.rm = TRUE)) %>% 
  ggplot(aes(x = quarter, y = avg_stays, group = simd))+
  geom_line(aes(colour = simd))+
  theme(axis.text.x = element_text(angle = 45, hjust = 1))+
  scale_y_continuous(labels = scales::comma)
`summarise()` has grouped output by 'quarter'. You can override using the `.groups` argument.

NA
NA
# plot avg stay length for most and least deprived for emergency unpatients
simd %>% 
  filter(admission_type == "Emergency Inpatients", simd %in% c(1,5)) %>% 
  drop_na(simd) %>% 
  group_by(quarter,simd) %>% 
  summarise(avg_stay_length = mean(average_length_of_stay, na.rm = TRUE)) %>% 
  ggplot(aes(x = quarter, y = avg_stay_length)) +
  geom_line(aes(colour = simd, group = simd))+
  theme(axis.text.x = element_text(angle = 45, hjust = 1))
`summarise()` has grouped output by 'quarter'. You can override using the `.groups` argument.

NA
speciality <- read_csv("raw_data/non_covid_raw_data/inpatient_and_daycase_by_nhs_board_of_treatment_and_specialty.csv") %>% janitor::clean_names()
Rows: 95270 Columns: 18── Column specification ─────────────────────────────────────────
Delimiter: ","
chr (12): Quarter, QuarterQF, HB, HBQF, Location, LocationQF,...
dbl  (6): Episodes, LengthOfEpisode, AverageLengthOfEpisode, ...
ℹ Use `spec()` to retrieve the full column specification for this data.
ℹ Specify the column types or set `show_col_types = FALSE` to quiet this message.
speciality %>% 
  count(admission_type)

speciality %>% 
  count(hb)

speciality %>% 
  count(location)

speciality %>% 
  count(specialty_name)

# add averages 
speciality_averages <- speciality %>% 
  group_by(quarter, admission_type) %>% 
  mutate(avg_length_spell= mean(average_length_of_spell, na.rm = TRUE),
         avg_length_episode = mean(average_length_of_episode, na.rm = TRUE)) %>% 
  ungroup()


speciality_averages %>% 
  group_by(quarter, admission_type) %>% 
  slice(1) %>% 
  ggplot(aes(x = quarter, y = average_length_of_episode, group = admission_type)) + 
  geom_line(aes(colour = admission_type))+
  theme(axis.text.x = element_text(angle = 45, hjust = 1))

ae_wait_times <- read_csv("raw_data/non_covid_raw_data/monthly_ae_waitingtimes_202206.csv") %>% janitor::clean_names()
Rows: 15837 Columns: 25── Column specification ─────────────────────────────────────────
Delimiter: ","
chr (13): Country, HBT, TreatmentLocation, DepartmentType, Nu...
dbl (12): Month, NumberOfAttendancesAggregate, NumberOfAttend...
ℹ Use `spec()` to retrieve the full column specification for this data.
ℹ Specify the column types or set `show_col_types = FALSE` to quiet this message.
#glimpse(ae_wait_times)


#make a date and year column with the first date of every month
ae_wait_times <- ae_wait_times %>% 
  mutate(date = ym(month), .after = month,
         year = year(date))

#make a percent column with percent of patients meeting the 4hr target time
ae_wait_times <- ae_wait_times %>% 
  mutate(percent_4hr_target_achieved = (number_meeting_target_aggregate/number_of_attendances_aggregate)*100) %>% 
  #add an 8hr one - not currently used
mutate(percent_seen_within_8hr = ((number_of_attendances_aggregate-attendance_greater8hrs)/number_of_attendances_aggregate)*100)
# draw percentage of 4 hour wait for all years
for_plotly <- ae_wait_times %>% 
  filter(department_type == "Emergency Department") %>% 
  group_by(date, department_type) %>% 
  summarise(avg_4hr_target_made = mean(percent_4hr_target_achieved)) %>% 
  ggplot(aes(x = date, y = avg_4hr_target_made))+
  geom_line(aes(colour = department_type))+
  scale_x_date(date_breaks = "6 months", date_labels =  "%b %Y")+
  theme(axis.text.x = element_text(angle = 90, hjust = 1, size =7))+
  geom_smooth()+
  geom_vline(xintercept = as.numeric(as.Date("2008-01-01")), linetype=4)+
  geom_vline(xintercept = as.numeric(as.Date("2009-01-01")), linetype=4)+
  geom_vline(xintercept = as.numeric(as.Date("2010-01-01")), linetype=4)+
  geom_vline(xintercept = as.numeric(as.Date("2011-01-01")), linetype=4)+
  geom_vline(xintercept = as.numeric(as.Date("2012-01-01")), linetype=4)+
  geom_vline(xintercept = as.numeric(as.Date("2013-01-01")), linetype=4)+
  geom_vline(xintercept = as.numeric(as.Date("2014-01-01")), linetype=4)+
  geom_vline(xintercept = as.numeric(as.Date("2015-01-01")), linetype=4)+
  geom_vline(xintercept = as.numeric(as.Date("2016-01-01")), linetype=4)+
  geom_vline(xintercept = as.numeric(as.Date("2017-01-01")), linetype=4)+
  geom_vline(xintercept = as.numeric(as.Date("2018-01-01")), linetype=4)+
  geom_vline(xintercept = as.numeric(as.Date("2019-01-01")), linetype=4)+
  geom_vline(xintercept = as.numeric(as.Date("2020-01-01")), linetype=4)+
  labs(title = "percentage of A&E departments meeting the 4 hr target turnaround for patients",
       subtitle = "added in vertical lines for January to help")
`summarise()` has grouped output by 'date'. You can override using the `.groups` argument.
ggplotly(for_plotly)
`geom_smooth()` using method = 'loess' and formula 'y ~ x'
Warning: `gather_()` was deprecated in tidyr 1.2.0.
Please use `gather()` instead.
# 4hr wait by health board for all years facet wrapped
ae_wait_times %>% 
  filter(department_type == "Emergency Department") %>% 
  group_by(date, hbt) %>% 
  mutate(avg_4hr_target_made = mean(percent_4hr_target_achieved)) %>% 
  slice(1) %>%  
  ggplot(aes(x = date, y = avg_4hr_target_made))+
  geom_line(aes(colour = hbt))+
  scale_x_date(date_breaks = "6 months", date_labels =  "%b %Y")+
  theme(axis.text.x = element_text(angle = 90, hjust = 1, size =7))+
  geom_smooth()+
  geom_vline(xintercept = as.numeric(as.Date("2008-01-01")), linetype=4)+
  geom_vline(xintercept = as.numeric(as.Date("2009-01-01")), linetype=4)+
  geom_vline(xintercept = as.numeric(as.Date("2010-01-01")), linetype=4)+
  geom_vline(xintercept = as.numeric(as.Date("2011-01-01")), linetype=4)+
  geom_vline(xintercept = as.numeric(as.Date("2012-01-01")), linetype=4)+
  geom_vline(xintercept = as.numeric(as.Date("2013-01-01")), linetype=4)+
  geom_vline(xintercept = as.numeric(as.Date("2014-01-01")), linetype=4)+
  geom_vline(xintercept = as.numeric(as.Date("2015-01-01")), linetype=4)+
  geom_vline(xintercept = as.numeric(as.Date("2016-01-01")), linetype=4)+
  geom_vline(xintercept = as.numeric(as.Date("2017-01-01")), linetype=4)+
  geom_vline(xintercept = as.numeric(as.Date("2018-01-01")), linetype=4)+
  geom_vline(xintercept = as.numeric(as.Date("2019-01-01")), linetype=4)+
  facet_wrap(~ hbt)

target_2007 <- ae_wait_times %>%
  group_by(year, hbt) %>% 
  summarise(ae_4hr_target_achieved = mean(percent_4hr_target_achieved, na.rm = TRUE)) %>% 
  filter(year == 2007) %>% 
  rename(ae_target_2007 = ae_4hr_target_achieved) %>% 
  ungroup() %>% 
  select(hbt,ae_target_2007)
`summarise()` has grouped output by 'year'. You can override using the `.groups` argument.
target_2008 <- ae_wait_times %>%
  group_by(year, hbt) %>% 
  summarise(ae_4hr_target_achieved = mean(percent_4hr_target_achieved, na.rm = TRUE)) %>% 
  filter(year == 2008) %>% 
  rename(ae_target_2008 = ae_4hr_target_achieved) %>% 
  ungroup() %>% 
  select(hbt,ae_target_2008)
`summarise()` has grouped output by 'year'. You can override using the `.groups` argument.
target_2009 <- ae_wait_times %>%
  group_by(year, hbt) %>% 
  summarise(ae_4hr_target_achieved = mean(percent_4hr_target_achieved, na.rm = TRUE)) %>% 
  filter(year == 2009) %>% 
  rename(ae_target_2009 = ae_4hr_target_achieved) %>% 
  ungroup() %>% 
  select(hbt,ae_target_2009)
`summarise()` has grouped output by 'year'. You can override using the `.groups` argument.
target_2010 <- ae_wait_times %>%
  group_by(year, hbt) %>% 
  summarise(ae_4hr_target_achieved = mean(percent_4hr_target_achieved, na.rm = TRUE)) %>% 
  filter(year == 2010) %>% 
  rename(ae_target_2010 = ae_4hr_target_achieved) %>% 
  ungroup() %>% 
  select(hbt,ae_target_2010)
`summarise()` has grouped output by 'year'. You can override using the `.groups` argument.
target_2011 <- ae_wait_times %>%
  group_by(year, hbt) %>% 
  summarise(ae_4hr_target_achieved = mean(percent_4hr_target_achieved, na.rm = TRUE)) %>% 
  filter(year == 2011) %>% 
  rename(ae_target_2011 = ae_4hr_target_achieved) %>% 
  ungroup() %>% 
  select(hbt,ae_target_2011)
`summarise()` has grouped output by 'year'. You can override using the `.groups` argument.
target_2012 <- ae_wait_times %>%
  group_by(year, hbt) %>% 
  summarise(ae_4hr_target_achieved = mean(percent_4hr_target_achieved, na.rm = TRUE)) %>% 
  filter(year == 2012) %>% 
  rename(ae_target_2012 = ae_4hr_target_achieved) %>% 
  ungroup() %>% 
  select(hbt,ae_target_2012)
`summarise()` has grouped output by 'year'. You can override using the `.groups` argument.
target_2013 <- ae_wait_times %>%
  group_by(year, hbt) %>% 
  summarise(ae_4hr_target_achieved = mean(percent_4hr_target_achieved, na.rm = TRUE)) %>% 
  filter(year == 2013) %>% 
  rename(ae_target_2013 = ae_4hr_target_achieved) %>% 
  ungroup() %>% 
  select(hbt,ae_target_2013)
`summarise()` has grouped output by 'year'. You can override using the `.groups` argument.
target_2014 <- ae_wait_times %>%
  group_by(year, hbt) %>% 
  summarise(ae_4hr_target_achieved = mean(percent_4hr_target_achieved, na.rm = TRUE)) %>% 
  filter(year == 2014) %>% 
  rename(ae_target_2014 = ae_4hr_target_achieved) %>% 
  ungroup() %>% 
  select(hbt,ae_target_2014)
`summarise()` has grouped output by 'year'. You can override using the `.groups` argument.
target_2015 <- ae_wait_times %>%
  group_by(year, hbt) %>% 
  summarise(ae_4hr_target_achieved = mean(percent_4hr_target_achieved, na.rm = TRUE)) %>% 
  filter(year == 2015) %>% 
  rename(ae_target_2015 = ae_4hr_target_achieved) %>% 
  ungroup() %>% 
  select(hbt,ae_target_2015)
`summarise()` has grouped output by 'year'. You can override using the `.groups` argument.
target_2016 <- ae_wait_times %>%
  group_by(year, hbt) %>% 
  summarise(ae_4hr_target_achieved = mean(percent_4hr_target_achieved, na.rm = TRUE)) %>% 
  filter(year == 2016) %>% 
  rename(ae_target_2016 = ae_4hr_target_achieved) %>% 
  ungroup() %>% 
  select(hbt,ae_target_2016)
`summarise()` has grouped output by 'year'. You can override using the `.groups` argument.
target_2017 <- ae_wait_times %>%
  group_by(year, hbt) %>% 
  summarise(ae_4hr_target_achieved = mean(percent_4hr_target_achieved, na.rm = TRUE)) %>% 
  filter(year == 2017) %>% 
  rename(ae_target_2017 = ae_4hr_target_achieved) %>% 
  ungroup() %>% 
  select(hbt,ae_target_2017)
`summarise()` has grouped output by 'year'. You can override using the `.groups` argument.
target_2018 <- ae_wait_times %>%
  group_by(year, hbt) %>% 
  summarise(ae_4hr_target_achieved = mean(percent_4hr_target_achieved, na.rm = TRUE)) %>% 
  filter(year == 2018) %>% 
  rename(ae_target_2018 = ae_4hr_target_achieved) %>% 
  ungroup() %>% 
  select(hbt,ae_target_2018)
`summarise()` has grouped output by 'year'. You can override using the `.groups` argument.
target_2019 <- ae_wait_times %>%
  group_by(year, hbt) %>% 
  summarise(ae_4hr_target_achieved = mean(percent_4hr_target_achieved, na.rm = TRUE)) %>% 
  filter(year == 2019) %>% 
  rename(ae_target_2019 = ae_4hr_target_achieved) %>% 
  ungroup() %>% 
  select(hbt,ae_target_2019)
`summarise()` has grouped output by 'year'. You can override using the `.groups` argument.
target_2020 <- ae_wait_times %>%
  group_by(year, hbt) %>% 
  summarise(ae_4hr_target_achieved = mean(percent_4hr_target_achieved, na.rm = TRUE)) %>% 
  filter(year == 2020) %>% 
  rename(ae_target_2020 = ae_4hr_target_achieved) %>% 
  ungroup() %>% 
  select(hbt,ae_target_2020)
`summarise()` has grouped output by 'year'. You can override using the `.groups` argument.
target_2021 <- ae_wait_times %>%
  group_by(year, hbt) %>% 
  summarise(ae_4hr_target_achieved = mean(percent_4hr_target_achieved, na.rm = TRUE)) %>% 
  filter(year == 2021) %>% 
  rename(ae_target_2021 = ae_4hr_target_achieved) %>% 
  ungroup() %>% 
  select(hbt,ae_target_2021)
`summarise()` has grouped output by 'year'. You can override using the `.groups` argument.
ggplot(data = scotland) +
geom_sf(fill = "green")+
ggrepel::geom_text_repel(aes(x = lat , y = long, label = paste(HBCode, HBName, sep = "\n")), min.segment.length = 0.05,size = 2, color = "black", fontface = "bold") +
  theme_void()


library(sf)

scotland_smaller <- scotland %>% # make a smaller version for performance issues
  st_simplify(TRUE, dTolerance = 2000)
#fixes problems caused by above 
scotland_smaller <- sf::st_cast(scotland_smaller, "MULTIPOLYGON")


# 
#   fig <- ggplotly(
#     ggplot(scotland)+
#   geom_sf(aes(fill = HBName))
# )
#   fig

  
  p <- ggplot(scotland_smaller) + 
  geom_sf(aes(fill = HBName, text = paste("<b>", HBName, "</b>\n", HBCode)))+
    theme_void()
Warning: Ignoring unknown aesthetics: text
  p %>%
  ggplotly(tooltip = "text") %>%
  style(hoverlabel = list(bgcolor = "white"), hoveron = "fill")
NA
NA

covid_age_sex <- read_csv("raw_data/covid_raw_data/hospital_admissions_hb_agesex_20220302.csv")
Rows: 43516 Columns: 12── Column specification ─────────────────────────────────────────
Delimiter: ","
chr (8): HB, HBQF, AgeGroup, AgeGroupQF, Sex, SexQF, Admissio...
dbl (4): WeekEnding, NumberAdmissions, Average20182019, Perce...
ℹ Use `spec()` to retrieve the full column specification for this data.
ℹ Specify the column types or set `show_col_types = FALSE` to quiet this message.
head(covid_age_sex)
library(leaflet)

data(quakes)

# Show first 20 rows from the `quakes` dataset
leaflet(data = quakes[1:20,]) %>% addTiles() %>%
  addMarkers(~long, ~lat, popup = ~as.character(mag))
LS0tDQp0aXRsZTogIlIgTm90ZWJvb2siDQpvdXRwdXQ6IGh0bWxfbm90ZWJvb2sNCi0tLQ0KYGBge3J9DQpsaWJyYXJ5KHRpZHl2ZXJzZSkNCmxpYnJhcnkoc2NhbGVzKQ0KbGlicmFyeShwbG90bHkpDQpsaWJyYXJ5KGx1YnJpZGF0ZSkNCmBgYA0KIyBCZWRzDQpgYGB7cn0NCmJlZHMgPC0gcmVhZF9jc3YoInJhd19kYXRhL25vbl9jb3ZpZF9yYXdfZGF0YS9iZWRzX2J5X25oc19ib2FyZF9vZl90cmVhdG1lbnRfYW5kX3NwZWNpYWx0eS5jc3YiKSAlPiUgamFuaXRvcjo6Y2xlYW5fbmFtZXMoKQ0KYGBgDQoNCmBgYHtyfQ0KYmVkcyAlPiUgDQogIGZpbHRlcihzcGVjaWFsdHlfbmFtZSA9PSAiQWxsIEFjdXRlIikgJT4lIA0KICBnZ3Bsb3QoYWVzKHggPSBxdWFydGVyLCB5ID0gcGVyY2VudGFnZV9vY2N1cGFuY3kpKSsNCiAgZ2VvbV9saW5lKGFlcyhjb2xvdXIgPSBoYiksIGdyb3VwID0gMSkrDQogIGZhY2V0X3dyYXAofiBoYikNCmBgYA0KDQpgYGB7cn0NCmJlZHMgPC0gYmVkcyAlPiUgDQogIHNlbGVjdCgtYygyLDQsNiw4LDEwLDEyLDE0LDE2LDE4LDIwKSkgJT4lIA0KICBmaWx0ZXIoIWhiICVpbiUgYygiU0IwODAxIiwgIlM5MjAwMDAwMyIpKQ0KDQpiZWRzIDwtIGJlZHMgJT4lIA0KICBmaWx0ZXIoIWhiICVpbiUgYygiU0IwODAxIiwgIlM5MjAwMDAwMyIpKSAlPiUgDQogIGZpbHRlcihoYiA9PSBsb2NhdGlvbikNCiAgDQojIGJlZHMgJT4lIA0KIyBjb3VudChzcGVjaWFsdHlfbmFtZSkNCg0KYV9lX2JlZHMgPC0gYmVkcyAlPiUgDQogIGZpbHRlcihzcGVjaWFsdHlfbmFtZSA9PSAiQWNjaWRlbnQgJiBFbWVyZ2VuY3kiKQ0KYGBgDQoNCmBgYHtyfQ0KIyBhJmUgcGVyY2VudGFnZSBvY2N1cGFuY3kgYnkgaGIgb3ZlciB0aW1lDQphX2VfYmVkcyAlPiUgDQogIGdyb3VwX2J5KHF1YXJ0ZXIsIGhiKSAlPiUgDQogIHN1bW1hcmlzZShtZWFuX3BlcmNfb2NjID0gbWVhbihwZXJjZW50YWdlX29jY3VwYW5jeSwgbmEucm0gPSBUUlVFKSkgJT4lIA0KICBnZ3Bsb3QoYWVzKHggPSBxdWFydGVyLCB5ID0gbWVhbl9wZXJjX29jYykpKw0KICBnZW9tX2xpbmUoYWVzKGdyb3VwID0gaGIsIGNvbG91ciA9IGhiKSkrDQogIGdlb21fcG9pbnQoKSsNCiAgdGhlbWUoYXhpcy50ZXh0LnggPSBlbGVtZW50X3RleHQoYW5nbGUgPSA0NSkpDQoNCg0KYV9lX2JlZHMgJT4lIA0KICBjb3VudChoYikNCmBgYA0KDQpgYGB7cn0NCiNhbGwgdGltZSBiZWQgb2NjdXBhbmN5IHBlcmNlbnRhZ2UgZm9yIGhlYWx0aCBib2FyZHMNCmFfZV9iZWRzICU+JSANCiAgZ3JvdXBfYnkocXVhcnRlciwgaGIpICU+JSANCiAgc3VtbWFyaXNlKG1lYW5fcGVyY19vY2MgPSBtZWFuKHBlcmNlbnRhZ2Vfb2NjdXBhbmN5LCBuYS5ybSA9IFRSVUUpKSAlPiUgDQp1bmdyb3VwKCkgJT4lIA0KICBncm91cF9ieShoYikgJT4lIA0KICBzdW1tYXJpc2UoYXZnX3Blcl9vY2NfYWxsX3RpbWUgPSBtZWFuKG1lYW5fcGVyY19vY2MpKSAlPiUgDQogIGFycmFuZ2UoZGVzYyhhdmdfcGVyX29jY19hbGxfdGltZSkpICU+JSANCiAgZ2dwbG90KGFlcyh4ID0gaGIsIHkgPSBhdmdfcGVyX29jY19hbGxfdGltZSkpKw0KICBnZW9tX2NvbCgpKw0KICBnZW9tX3BvaW50KCkrDQogIHRoZW1lKGF4aXMudGV4dC54ID0gZWxlbWVudF90ZXh0KGFuZ2xlID0gNDUpKQ0KDQpgYGANCg0KDQpgYGB7cn0NCiMgd29ya291dCB0aGUgdGVuIGxhcmdlc3QgDQp0ZW5fbGFyZ2VzdF9zcGVjaWFsaXRpZXMgPC0gYmVkcyAlPiUNCiAgZ3JvdXBfYnkoc3BlY2lhbHR5X25hbWUpICU+JSANCiAgc3VtbWFyaXNlKG1lYW5fYXZhaWxfc3RhZmZlZF9iZWRzID0gbWVhbihhdmVyYWdlX2F2YWlsYWJsZV9zdGFmZmVkX2JlZHMpKSAlPiUgDQogIGFycmFuZ2UoZGVzYyhtZWFuX2F2YWlsX3N0YWZmZWRfYmVkcykpICU+JSANCiAgc2xpY2VfbWF4KG1lYW5fYXZhaWxfc3RhZmZlZF9iZWRzLCBuPTEwKSAlPiUgDQogIHNlbGVjdCgxKSAlPiUgDQogIHB1bGwoKQ0KDQoNCg0KIyBiZWQgcGVyY2VudGFnZSBhdmFpbGFibGl0eSBmb3IgdG9wIHRlbiBsYXJnZXN0IHNwZWNpYWxpdGllcw0KYmVkcyAlPiUNCiAgZmlsdGVyKHNwZWNpYWx0eV9uYW1lICVpbiUgdGVuX2xhcmdlc3Rfc3BlY2lhbGl0aWVzKSAlPiUgDQogIGdyb3VwX2J5KHF1YXJ0ZXIsIHNwZWNpYWx0eV9uYW1lKSAlPiUNCiAgc3VtbWFyaXNlKG1lYW5fcGVyY19vY2MgPSBtZWFuKHBlcmNlbnRhZ2Vfb2NjdXBhbmN5KSkgJT4lIA0KICBnZ3Bsb3QoYWVzKHggPSBxdWFydGVyLCB5ID0gbWVhbl9wZXJjX29jYykpKw0KICBnZW9tX2xpbmUoYWVzKGNvbG91ciA9IHNwZWNpYWx0eV9uYW1lLCBncm91cCA9IHNwZWNpYWx0eV9uYW1lKSkrDQogIGdlb21fcG9pbnQoKSsNCiAgdGhlbWUoYXhpcy50ZXh0LnggPSBlbGVtZW50X3RleHQoYW5nbGUgPSA0NSwgaGp1c3QgPSAxKSkNCiAgDQojIGJlZCBwZXJjZW50YWdlIGF2YWlsYWJsaXR5IGZvciBhbGwgYWN1dGUNCiAgYmVkcyAlPiUNCiAgZmlsdGVyKHNwZWNpYWx0eV9uYW1lID09ICJBbGwgQWN1dGUiKSAlPiUgDQogIGdyb3VwX2J5KHF1YXJ0ZXIsIHNwZWNpYWx0eV9uYW1lKSAlPiUNCiAgc3VtbWFyaXNlKG1lYW5fcGVyY19vY2MgPSBtZWFuKHBlcmNlbnRhZ2Vfb2NjdXBhbmN5KSkgJT4lIA0KICBnZ3Bsb3QoYWVzKHggPSBxdWFydGVyLCB5ID0gbWVhbl9wZXJjX29jYykpKw0KICBnZW9tX2xpbmUoYWVzKGNvbG91ciA9IHNwZWNpYWx0eV9uYW1lLCBncm91cCA9IHNwZWNpYWx0eV9uYW1lKSkrDQogIGdlb21fcG9pbnQoKSsNCiAgdGhlbWUoYXhpcy50ZXh0LnggPSBlbGVtZW50X3RleHQoYW5nbGUgPSA0NSwgaGp1c3QgPSAxKSkNCiAgDQogICMgYmVkIHBlcmNlbnRhZ2UgYXZhaWxhYmlsaXR5IGZvciBpbnRlbnNpdmUgY2FyZQ0KYmVkcyAlPiUNCiAgZmlsdGVyKHNwZWNpYWx0eV9uYW1lID09ICJJbnRlbnNpdmUgQ2FyZSBNZWRpY2luZSIpICU+JSANCiAgZ3JvdXBfYnkocXVhcnRlciwgc3BlY2lhbHR5X25hbWUpICU+JQ0KICBzdW1tYXJpc2UobWVhbl9wZXJjX29jYyA9IG1lYW4ocGVyY2VudGFnZV9vY2N1cGFuY3kpKSAlPiUgDQogIGdncGxvdChhZXMoeCA9IHF1YXJ0ZXIsIHkgPSBtZWFuX3BlcmNfb2NjKSkrDQogIGdlb21fbGluZShhZXMoY29sb3VyID0gc3BlY2lhbHR5X25hbWUsIGdyb3VwID0gc3BlY2lhbHR5X25hbWUpKSsNCiAgZ2VvbV9wb2ludCgpKw0KICB0aGVtZShheGlzLnRleHQueCA9IGVsZW1lbnRfdGV4dChhbmdsZSA9IDQ1LCBoanVzdCA9IDEpKQ0KICANCiAgDQogIA0KYGBgDQoNCg0KDQpgYGB7cn0NCmFfZV9iZWRzICU+JSANCiAgZ3JvdXBfYnkocXVhcnRlcikgJT4lIA0KICBzdW1tYXJpc2UobWVhbl9hdmFpbGFibGVfYmVkcyA9IG1lYW4oYXZlcmFnZV9hdmFpbGFibGVfc3RhZmZlZF9iZWRzLCBuYS5ybSA9IFRSVUUpKSAlPiUgDQogIGdncGxvdChhZXMoeCA9IHF1YXJ0ZXIsIHkgPSBtZWFuX2F2YWlsYWJsZV9iZWRzKSkrDQogIGdlb21fbGluZShncm91cCA9IjEiKSsNCiAgZ2VvbV9wb2ludCgpKw0KICB0aGVtZShheGlzLnRleHQueCA9IGVsZW1lbnRfdGV4dChhbmdsZSA9IDQ1KSkNCmBgYA0KDQoNCi0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tDQoNCiMgYWdlIHNleA0KYGBge3J9DQphZ2Vfc2V4IDwtIHJlYWRfY3N2KCJyYXdfZGF0YS9ub25fY292aWRfcmF3X2RhdGEvaW5wYXRpZW50X2FuZF9kYXljYXNlX2J5X25oc19ib2FyZF9vZl90cmVhdG1lbnRfYWdlX2FuZF9zZXguY3N2IikgJT4lIGphbml0b3I6OmNsZWFuX25hbWVzKCkNCg0KaGVhbHRoX2JvYXJkX25hbWVzIDwtIHJlYWRfY3N2KCJyYXdfZGF0YS9ub25fY292aWRfcmF3X2RhdGEvaGVhbHRoX2JvYXJkX25hbWVzLmNzdiIpDQoNCmFnZV9zZXggJT4lIA0KICBjb3VudChoYikNCg0Kc2Vhc29uX2FnZV9zZXggPC0gYWdlX3NleCAlPiUgDQogIG11dGF0ZShkYXRlID0geXEocXVhcnRlciksDQogICAgICAgICB5ZWFyID0geWVhcihkYXRlKSwNCiAgICAgICAgIG1vbnRoID0gbW9udGgoZGF0ZSwgbGFiZWwgPSBUUlVFLCBhYmJyID0gRkFMU0UpLA0KICAgICAgICAgc2Vhc29uID0gY2FzZV93aGVuKA0KICAgICAgICAgICBzdHJfZGV0ZWN0KG1vbnRoLCAiRGVjZW1iZXIiKSB+ICJXaW50ZXIiLA0KICAgICAgICAgICBzdHJfZGV0ZWN0KG1vbnRoLCAiSmFudWFyeSIpIH4gIldpbnRlciIsDQogICAgICAgICAgIHN0cl9kZXRlY3QobW9udGgsICJGZWJydWFyeSIpIH4gIldpbnRlciIsDQogICAgICAgICAgIHN0cl9kZXRlY3QobW9udGgsICJNYXJjaCIpIH4gIlNwcmluZyIsDQogICAgICAgICAgIHN0cl9kZXRlY3QobW9udGgsICJBcHJpbCIpIH4gIlNwcmluZyIsDQogICAgICAgICAgIHN0cl9kZXRlY3QobW9udGgsICJNYXkiKSB+ICJTcHJpbmciLA0KICAgICAgICAgICBzdHJfZGV0ZWN0KG1vbnRoLCAiSnVuZSIpIH4gIlN1bW1lciIsDQogICAgICAgICAgIHN0cl9kZXRlY3QobW9udGgsICJKdWx5IikgfiAiU3VtbWVyIiwNCiAgICAgICAgICAgc3RyX2RldGVjdChtb250aCwgIkF1Z3VzdCIpIH4gIlN1bW1lciIsDQogICAgICAgICAgIHN0cl9kZXRlY3QobW9udGgsICJTZXB0ZW1iZXIiKSB+ICJBdXR1bW4iLA0KICAgICAgICAgICBzdHJfZGV0ZWN0KG1vbnRoLCAiT2N0b2JlciIpIH4gIkF1dHVtbiIsDQogICAgICAgICAgIHN0cl9kZXRlY3QobW9udGgsICJOb3ZlbWJlciIpIH4gIkF1dHVtbiIpLA0KICAgICAgICAgc2Vhc29uID0gZmFjdG9yKHNlYXNvbiwgb3JkZXIgPSBUUlVFKSkgDQpgYGANCmBgYHtyfQ0KDQpsaWJyYXJ5KGx1YnJpZGF0ZSkNCmxpYnJhcnkoem9vKQ0KDQojIGNoYW5nZSBxdWFydGVyIGNvbHVtbiBpbnRvIHRoZSBkYXRlIGF0IHRoZSBzdGFydCBvZiBlYWNoIHF1YXJ0ZXINCiBhZ2Vfc2V4IDwtICBhZ2Vfc2V4ICU+JSANCiAgICBtdXRhdGUocXVhcnRlciA9IHlxKHF1YXJ0ZXIpKQ0KDQogIyBzaG93cyB0aGUgdG90YWwgbGVuZ3RoIG9mIHN0YXkgYnkgYWdlIGJyYWNrZXQgZm9yIGVtZXJnZW5jeSBpbnBhdGllbnRzDQphZ2Vfc2V4ICU+JSANCiAgZmlsdGVyKGFkbWlzc2lvbl90eXBlID09ICJFbWVyZ2VuY3kgSW5wYXRpZW50cyIpICU+JSANCiAgZ3JvdXBfYnkocXVhcnRlciwgYWdlKSAlPiUgDQogIHN1bW1hcmlzZSh0b3RhbF9sZW5ndGhfb2Zfc3RheSA9IHN1bShsZW5ndGhfb2Zfc3RheSkpICU+JSANCiAgZ2dwbG90KGFlcyh4ID0gcXVhcnRlciwgeSA9IHRvdGFsX2xlbmd0aF9vZl9zdGF5KSkrDQogIGdlb21fbGluZShhZXMoY29sb3VyID0gYWdlKSkrDQogIHRoZW1lKGF4aXMudGV4dC54ID0gZWxlbWVudF90ZXh0KGFuZ2xlID0gNDUpKQ0KDQoNCmBgYA0KDQoNCg0KDQoNCmBgYHtyfQ0KYWdlX3NleCAlPiUgDQpjb3VudChhZG1pc3Npb25fdHlwZSkNCg0KYWdlX3NleA0KYGBgDQoNCmBgYHtyfQ0KICMgc2hvd3MgdGhlIHRvdGFsIGxlbmd0aCBvZiBzdGF5IGJ5IGFnZSBicmFja2V0IGZvciBlbGVjdGl2ZSBpbnBhdGllbnRzDQphZ2Vfc2V4ICU+JSANCiAgZmlsdGVyKGFkbWlzc2lvbl90eXBlID09ICJFbGVjdGl2ZSBJbnBhdGllbnRzIikgJT4lIA0KICBncm91cF9ieShxdWFydGVyLCBhZ2UpICU+JSANCiAgc3VtbWFyaXNlKHRvdGFsX2xlbmd0aF9vZl9zdGF5ID0gc3VtKGxlbmd0aF9vZl9zdGF5KSkgJT4lIA0KICBnZ3Bsb3QoYWVzKHggPSBxdWFydGVyLCB5ID0gdG90YWxfbGVuZ3RoX29mX3N0YXkpKSsNCiAgZ2VvbV9saW5lKGFlcyhjb2xvdXIgPSBhZ2UpKSsNCiAgdGhlbWUoYXhpcy50ZXh0LnggPSBlbGVtZW50X3RleHQoYW5nbGUgPSA0NSkpDQpgYGANCg0KYGBge3J9DQogIyBzaG93cyB0aGUgbWVhbiBsZW5ndGggb2Ygc3RheSBieSBhZ2UgYnJhY2tldCBmb3IgZW1lcmdlbmN5IGlucGF0aWVudHMNCiNjYW4gZmFjZXQgYnkgc2V4IGFsc28gaWYgcmVxdWlyZWQNCmFnZV9zZXggJT4lIA0KICBmaWx0ZXIoYWRtaXNzaW9uX3R5cGUgPT0gIkVtZXJnZW5jeSBJbnBhdGllbnRzIikgJT4lIA0KICBncm91cF9ieShxdWFydGVyLCBhZ2UpICU+JSANCiAgc3VtbWFyaXNlKGF2Z19sZW5ndGhfb2Zfc3RheSA9IG1lYW4oYXZlcmFnZV9sZW5ndGhfb2Zfc3RheSwgbmEucm0gPSBUUlVFKSkgJT4lIA0KICBnZ3Bsb3QoYWVzKHggPSBxdWFydGVyLCB5ID0gYXZnX2xlbmd0aF9vZl9zdGF5KSkrDQogIGdlb21fbGluZShhZXMoY29sb3VyID0gYWdlLCBncm91cCA9IGFnZSkpKw0KICB0aGVtZShheGlzLnRleHQueCA9IGVsZW1lbnRfdGV4dChhbmdsZSA9IDQ1KSkNCiAgI2ZhY2V0X3dyYXAoIH4gc2V4KQ0KDQogIyBzaG93cyB0aGUgbWVhbiBsZW5ndGggb2Ygc3RheSBieSBhZ2UgYnJhY2tldCBmb3IgZWxlY3RpdmUgaW5wYXRpZW50cw0KIyBjYW4gZmFjZXQgYnkgc2V4IA0KYWdlX3NleCAlPiUgDQogIGZpbHRlcihhZG1pc3Npb25fdHlwZSA9PSAiRWxlY3RpdmUgSW5wYXRpZW50cyIpICU+JSANCiAgZ3JvdXBfYnkocXVhcnRlciwgYWdlKSAlPiUgDQogIHN1bW1hcmlzZShhdmdfbGVuZ3RoX29mX3N0YXkgPSBtZWFuKGF2ZXJhZ2VfbGVuZ3RoX29mX3N0YXksIG5hLnJtID0gVFJVRSkpICU+JSANCiAgZ2dwbG90KGFlcyh4ID0gcXVhcnRlciwgeSA9IGF2Z19sZW5ndGhfb2Zfc3RheSkpKw0KICBnZW9tX2xpbmUoYWVzKGNvbG91ciA9IGFnZSwgZ3JvdXAgPSBhZ2UpKSsNCiAgdGhlbWUoYXhpcy50ZXh0LnggPSBlbGVtZW50X3RleHQoYW5nbGUgPSA0NSkpDQojICAgI2ZhY2V0X3dyYXAofiBzZXgpDQpgYGANCg0KYGBge3J9DQphZ2Vfc2V4ICU+JSANCiAgbXV0YXRlKGRhdGUgPSB5cShxdWFydGVyKSwNCiAgICAgICAgIHllYXIgPSB5ZWFyKGRhdGUpLA0KICAgICAgICAgbW9udGggPSBtb250aChkYXRlLCBsYWJlbCA9IFRSVUUsIGFiYnIgPSBGQUxTRSksDQogICAgICAgICBzZWFzb24gPSBjYXNlX3doZW4oDQogICAgICAgICAgIHN0cl9kZXRlY3QobW9udGgsICJEZWNlbWJlciIpIH4gIldpbnRlciIsDQogICAgICAgICAgIHN0cl9kZXRlY3QobW9udGgsICJKYW51YXJ5IikgfiAiV2ludGVyIiwNCiAgICAgICAgICAgc3RyX2RldGVjdChtb250aCwgIkZlYnJ1YXJ5IikgfiAiV2ludGVyIiwNCiAgICAgICAgICAgc3RyX2RldGVjdChtb250aCwgIk1hcmNoIikgfiAiU3ByaW5nIiwNCiAgICAgICAgICAgc3RyX2RldGVjdChtb250aCwgIkFwcmlsIikgfiAiU3ByaW5nIiwNCiAgICAgICAgICAgc3RyX2RldGVjdChtb250aCwgIk1heSIpIH4gIlNwcmluZyIsDQogICAgICAgICAgIHN0cl9kZXRlY3QobW9udGgsICJKdW5lIikgfiAiU3VtbWVyIiwNCiAgICAgICAgICAgc3RyX2RldGVjdChtb250aCwgIkp1bHkiKSB+ICJTdW1tZXIiLA0KICAgICAgICAgICBzdHJfZGV0ZWN0KG1vbnRoLCAiQXVndXN0IikgfiAiU3VtbWVyIiwNCiAgICAgICAgICAgc3RyX2RldGVjdChtb250aCwgIlNlcHRlbWJlciIpIH4gIkF1dHVtbiIsDQogICAgICAgICAgIHN0cl9kZXRlY3QobW9udGgsICJPY3RvYmVyIikgfiAiQXV0dW1uIiwNCiAgICAgICAgICAgc3RyX2RldGVjdChtb250aCwgIk5vdmVtYmVyIikgfiAiQXV0dW1uIiksDQogICAgICAgICBzZWFzb24gPSBmYWN0b3Ioc2Vhc29uLCBvcmRlciA9IFRSVUUpKSANCmBgYA0KDQoNCmBgYHtyfQ0KDQphZ2Vfc2V4ICU+JSANCiAgZmlsdGVyKGFkbWlzc2lvbl90eXBlID09ICJFbWVyZ2VuY3kgSW5wYXRpZW50cyIpICU+JSANCiAgZ3JvdXBfYnkocXVhcnRlciwgc2V4KSAlPiUgDQogIHN1bW1hcmlzZShhdmdfbGVuZ3RoX29mX3N0YXkgPSBtZWFuKGF2ZXJhZ2VfbGVuZ3RoX29mX3N0YXksIG5hLnJtID0gVFJVRSkpICU+JSANCiAgZ2dwbG90KGFlcyh4ID0gcXVhcnRlciwgeSA9IGF2Z19sZW5ndGhfb2Zfc3RheSkpKw0KICBnZW9tX2xpbmUoYWVzKGNvbG91ciA9IHNleCwgZ3JvdXAgPSBzZXgpKSsNCiAgdGhlbWUoYXhpcy50ZXh0LnggPSBlbGVtZW50X3RleHQoYW5nbGUgPSA0NSkpKw0KICBnZW9tX3Ntb290aChhZXMoY29sb3VyID0gc2V4LCBncm91cCA9IHNleCksIHNlID0gRkFMU0UpKw0KICBnZW9tX3BvaW50KGFlcyhjb2xvdXIgPSBzZXgpLCBzaXplID0gMC41KSsNCiAgbGFicyh0aXRsZSA9ICJFbWVyZ2VuY3kgSW5wYXRpZW50IGJ5IGdlbmRlciBhbmQgYXZlcmFnZSBsZW5ndGggb2Ygc3RheSIpDQoNCg0KYWdlX3NleCAlPiUgDQogIGZpbHRlcihhZG1pc3Npb25fdHlwZSA9PSAiRWxlY3RpdmUgSW5wYXRpZW50cyIpICU+JSANCiAgZ3JvdXBfYnkocXVhcnRlciwgc2V4KSAlPiUgDQogIHN1bW1hcmlzZShhdmdfbGVuZ3RoX29mX3N0YXkgPSBtZWFuKGF2ZXJhZ2VfbGVuZ3RoX29mX3N0YXksIG5hLnJtID0gVFJVRSkpICU+JSANCiAgZ2dwbG90KGFlcyh4ID0gcXVhcnRlciwgeSA9IGF2Z19sZW5ndGhfb2Zfc3RheSkpKw0KICBnZW9tX2xpbmUoYWVzKGNvbG91ciA9IHNleCwgZ3JvdXAgPSBzZXgpKSsNCiAgdGhlbWUoYXhpcy50ZXh0LnggPSBlbGVtZW50X3RleHQoYW5nbGUgPSA0NSkpKw0KICBnZW9tX3Ntb290aChhZXMoY29sb3VyID0gc2V4LCBncm91cCA9IHNleCksIHNlID0gRkFMU0UpKw0KICBnZW9tX3BvaW50KGFlcyhjb2xvdXIgPSBzZXgpLCBzaXplID0gMC41KSsNCiAgbGFicyh0aXRsZSA9ICJFbGVjdGl2ZSBJbnBhdGllbnQgYnkgZ2VuZGVyIGFuZCBhdmVyYWdlIGxlbmd0aCBvZiBzdGF5IikNCg0KYGBgDQoNCg0KYGBge3J9DQojIGVtZXJnZW5jeSBpbnBhdGllbnQgYnkgYWdlIGFuZCBhdmcgbGVuZ3RoIG9mIHN0YXkNCmFnZV9zZXggJT4lIA0KICBmaWx0ZXIoYWRtaXNzaW9uX3R5cGUgPT0gIkVtZXJnZW5jeSBJbnBhdGllbnRzIikgJT4lIA0KICBncm91cF9ieShxdWFydGVyLCBhZ2UpICU+JSANCiAgc3VtbWFyaXNlKGF2Z19sZW5ndGhfb2Zfc3RheSA9IG1lYW4oYXZlcmFnZV9sZW5ndGhfb2Zfc3RheSwgbmEucm0gPSBUUlVFKSkgJT4lIA0KICBnZ3Bsb3QoYWVzKHggPSBxdWFydGVyLCB5ID0gYXZnX2xlbmd0aF9vZl9zdGF5KSkrDQogIGdlb21fbGluZShhZXMoY29sb3VyID0gYWdlLCBncm91cCA9IGFnZSkpKw0KICB0aGVtZShheGlzLnRleHQueCA9IGVsZW1lbnRfdGV4dChhbmdsZSA9IDQ1KSkrDQogICNnZW9tX3Ntb290aChhZXMoY29sb3VyID0gc2V4LCBncm91cCA9IHNleCksIHNlID0gRkFMU0UpKw0KICBnZW9tX3BvaW50KGFlcyhjb2xvdXIgPSBhZ2UpLCBzaXplID0gMC41KSsNCiAgbGFicyh0aXRsZSA9ICJFbWVyZ2VuY3kgaW5wYXRpZW50IGJ5IGFnZSBhbmQgYXZnIGxlbmd0aCBvZiBzdGF5IikNCg0KIyBlbGVjdGl2ZSBpbnBhdGllbnQgYnkgYWdlIGFuZCBhdmcgbGVuZ3RoIG9mIHN0YXkNCmFnZV9zZXggJT4lIA0KICBmaWx0ZXIoYWRtaXNzaW9uX3R5cGUgPT0gIkVsZWN0aXZlIElucGF0aWVudHMiKSAlPiUgDQogIGdyb3VwX2J5KHF1YXJ0ZXIsIGFnZSkgJT4lIA0KICBzdW1tYXJpc2UoYXZnX2xlbmd0aF9vZl9zdGF5ID0gbWVhbihhdmVyYWdlX2xlbmd0aF9vZl9zdGF5LCBuYS5ybSA9IFRSVUUpKSAlPiUgDQogIGdncGxvdChhZXMoeCA9IHF1YXJ0ZXIsIHkgPSBhdmdfbGVuZ3RoX29mX3N0YXkpKSsNCiAgZ2VvbV9saW5lKGFlcyhjb2xvdXIgPSBhZ2UsIGdyb3VwID0gYWdlKSkrDQogIHRoZW1lKGF4aXMudGV4dC54ID0gZWxlbWVudF90ZXh0KGFuZ2xlID0gNDUpKSsNCiAgI2dlb21fc21vb3RoKGFlcyhjb2xvdXIgPSBzZXgsIGdyb3VwID0gc2V4KSwgc2UgPSBGQUxTRSkrDQogIGdlb21fcG9pbnQoYWVzKGNvbG91ciA9IGFnZSksIHNpemUgPSAwLjUpKw0KICBsYWJzKHRpdGxlID0gIkVsZWN0aXZlIGlucGF0aWVudCBieSBhZ2UgYW5kIGF2ZyBsZW5ndGggb2Ygc3RheSIpDQpgYGANCg0KYGBge3J9DQojIGVtZXJnZW5jeSBpbnBhdGllbnQgYnkgYWdlIGFuZCBhdmcgbGVuZ3RoIG9mIGVwc2lvZGUNCmFnZV9zZXggJT4lIA0KICBmaWx0ZXIoYWRtaXNzaW9uX3R5cGUgPT0gIkVtZXJnZW5jeSBJbnBhdGllbnRzIikgJT4lIA0KICBncm91cF9ieShxdWFydGVyLCBhZ2UpICU+JSANCiAgc3VtbWFyaXNlKGF2Z19sZW5ndGhfb2ZfZXBpc29kZSA9IG1lYW4oYXZlcmFnZV9sZW5ndGhfb2ZfZXBpc29kZSwgbmEucm0gPSBUUlVFKSkgJT4lIA0KICBnZ3Bsb3QoYWVzKHggPSBxdWFydGVyLCB5ID0gYXZnX2xlbmd0aF9vZl9lcGlzb2RlKSkrDQogIGdlb21fbGluZShhZXMoY29sb3VyID0gYWdlLCBncm91cCA9IGFnZSkpKw0KICB0aGVtZShheGlzLnRleHQueCA9IGVsZW1lbnRfdGV4dChhbmdsZSA9IDQ1KSkrDQogICNnZW9tX3Ntb290aChhZXMoY29sb3VyID0gc2V4LCBncm91cCA9IHNleCksIHNlID0gRkFMU0UpKw0KICBnZW9tX3BvaW50KGFlcyhjb2xvdXIgPSBhZ2UpLCBzaXplID0gMC41KSsNCiAgbGFicyh0aXRsZSA9ICJFbWVyZ2VuY3kgaW5wYXRpZW50IGJ5IGFnZSBhbmQgYXZnIGxlbmd0aCBvZiBlcGlzb2RlIikNCg0KIyBlbGVjdGl2ZSBpbnBhdGllbnQgYnkgYWdlIGFuZCBhdmcgbGVuZ3RoIG9mIGVwaXNvZGUNCmFnZV9zZXggJT4lIA0KICBmaWx0ZXIoYWRtaXNzaW9uX3R5cGUgPT0gIkVsZWN0aXZlIElucGF0aWVudHMiKSAlPiUgDQogIGdyb3VwX2J5KHF1YXJ0ZXIsIGFnZSkgJT4lIA0KICBzdW1tYXJpc2UoYXZnX2xlbmd0aF9vZl9lcGlzb2RlID0gbWVhbihhdmVyYWdlX2xlbmd0aF9vZl9lcGlzb2RlLCBuYS5ybSA9IFRSVUUpKSAlPiUgDQogIGdncGxvdChhZXMoeCA9IHF1YXJ0ZXIsIHkgPSBhdmdfbGVuZ3RoX29mX2VwaXNvZGUpKSsNCiAgZ2VvbV9saW5lKGFlcyhjb2xvdXIgPSBhZ2UsIGdyb3VwID0gYWdlKSkrDQogIHRoZW1lKGF4aXMudGV4dC54ID0gZWxlbWVudF90ZXh0KGFuZ2xlID0gNDUpKSsNCiAgI2dlb21fc21vb3RoKGFlcyhjb2xvdXIgPSBzZXgsIGdyb3VwID0gc2V4KSwgc2UgPSBGQUxTRSkrDQogIGdlb21fcG9pbnQoYWVzKGNvbG91ciA9IGFnZSksIHNpemUgPSAwLjUpKw0KICBsYWJzKHRpdGxlID0gIkVsZWN0aXZlIGlucGF0aWVudCBieSBhZ2UgYW5kIGF2ZyBsZW5ndGggb2YgZXBpc29kZSIpDQpgYGANCg0KYGBge3J9DQojIGVtZXJnZW5jeSBpbnBhdGllbnQgYnkgYWdlIGFuZCBhdmcgZXBpc29kZXMNCmFnZV9zZXggJT4lIA0KICBmaWx0ZXIoYWRtaXNzaW9uX3R5cGUgPT0gIkVtZXJnZW5jeSBJbnBhdGllbnRzIikgJT4lIA0KICBncm91cF9ieShxdWFydGVyLCBhZ2UpICU+JSANCiAgc3VtbWFyaXNlKGF2Z19lcGlzb2RlcyA9IG1lYW4oZXBpc29kZXMsIG5hLnJtID0gVFJVRSkpICU+JSANCiAgZ2dwbG90KGFlcyh4ID0gcXVhcnRlciwgeSA9IGF2Z19lcGlzb2RlcykpKw0KICBnZW9tX2xpbmUoYWVzKGNvbG91ciA9IGFnZSwgZ3JvdXAgPSBhZ2UpKSsNCiAgdGhlbWUoYXhpcy50ZXh0LnggPSBlbGVtZW50X3RleHQoYW5nbGUgPSA0NSkpKw0KICAjZ2VvbV9zbW9vdGgoYWVzKGNvbG91ciA9IHNleCwgZ3JvdXAgPSBzZXgpLCBzZSA9IEZBTFNFKSsNCiAgZ2VvbV9wb2ludChhZXMoY29sb3VyID0gYWdlKSwgc2l6ZSA9IDAuNSkrDQogIGxhYnModGl0bGUgPSAiRW1lcmdlbmN5IGlucGF0aWVudCBieSBhZ2UgYW5kIGF2ZyBlcGlzb2RlcyIpDQoNCiMgZW1lcmdlbmN5IGlucGF0aWVudCBieSBhZ2UgYW5kIGF2ZyBlcGlzb2Rlcw0KYWdlX3NleCAlPiUgDQogIGZpbHRlcihhZG1pc3Npb25fdHlwZSA9PSAiRWxlY3RpdmUgSW5wYXRpZW50cyIpICU+JSANCiAgZ3JvdXBfYnkocXVhcnRlciwgYWdlKSAlPiUgDQogIHN1bW1hcmlzZShhdmdfZXBpc29kZXMgPSBtZWFuKGVwaXNvZGVzLCBuYS5ybSA9IFRSVUUpKSAlPiUgDQogIGdncGxvdChhZXMoeCA9IHF1YXJ0ZXIsIHkgPSBhdmdfZXBpc29kZXMpKSsNCiAgZ2VvbV9saW5lKGFlcyhjb2xvdXIgPSBhZ2UsIGdyb3VwID0gYWdlKSkrDQogIHRoZW1lKGF4aXMudGV4dC54ID0gZWxlbWVudF90ZXh0KGFuZ2xlID0gNDUpKSsNCiAgI2dlb21fc21vb3RoKGFlcyhjb2xvdXIgPSBzZXgsIGdyb3VwID0gc2V4KSwgc2UgPSBGQUxTRSkrDQogIGdlb21fcG9pbnQoYWVzKGNvbG91ciA9IGFnZSksIHNpemUgPSAwLjUpKw0KICBsYWJzKHRpdGxlID0gIkVsZWN0aXZlIGlucGF0aWVudCBieSBhZ2UgYW5kIGF2ZyBlcGlzb2RlcyIpDQpgYGANCg0KDQoNCmBgYHtyfQ0KIyBQbG90IGNvbXBhcmlzb24gb2YgRW1lcmdlbmN5IHZzIEVsZWN0aXZlIHN1Ym1pc3Npb25zDQphZ2Vfc2V4ICU+JSANCiAgZmlsdGVyKGFkbWlzc2lvbl90eXBlICVpbiUgYygiRW1lcmdlbmN5IElucGF0aWVudHMiLCAiRWxlY3RpdmUgSW5wYXRpZW50cyIpKSAlPiUgDQogIGdyb3VwX2J5KHF1YXJ0ZXIsIGFkbWlzc2lvbl90eXBlKSAlPiUgDQogIHN1bW1hcmlzZShhdmdfc3RheSA9IG1lYW4oYXZlcmFnZV9sZW5ndGhfb2Zfc3RheSwgbmEucm0gPSBUUlVFKSkgJT4lIA0KICBnZ3Bsb3QoYWVzKHggPSBxdWFydGVyLCB5ID0gYXZnX3N0YXksIGNvbG91ciA9IGFkbWlzc2lvbl90eXBlKSkrDQogIGdlb21fbGluZShhZXMoZ3JvdXAgPSBhZG1pc3Npb25fdHlwZSkpKw0KICBnZW9tX3BvaW50KCkrDQogIHRoZW1lKGF4aXMudGV4dC54ID0gZWxlbWVudF90ZXh0KGFuZ2xlID0gNDUsIGhqdXN0ID0gMSkpDQoNCmBgYA0KDQoNCmBgYHtyfQ0KIyBhdmdfc3RheSBieSBhZG1pc3Npb24gdHlwZQ0KYWdlX3NleCAlPiUgDQogIGdyb3VwX2J5KHF1YXJ0ZXIsIGFkbWlzc2lvbl90eXBlKSAlPiUgDQogIHN1bW1hcmlzZShhdmdfc3RheSA9IG1lYW4oYXZlcmFnZV9sZW5ndGhfb2Zfc3RheSwgbmEucm0gPSBUUlVFKSkgJT4lIA0KICBnZ3Bsb3QoYWVzKHggPSBxdWFydGVyLCB5ID0gYXZnX3N0YXksIGNvbG91ciA9IGFkbWlzc2lvbl90eXBlKSkrDQogIGdlb21fbGluZShhZXMoZ3JvdXAgPSBhZG1pc3Npb25fdHlwZSkpKw0KICBnZW9tX3BvaW50KCkrDQogIHRoZW1lKGF4aXMudGV4dC54ID0gZWxlbWVudF90ZXh0KGFuZ2xlID0gNDUsIGhqdXN0ID0gMSkpDQoNCg0KIyBhdmdfc3RheSBmb3IgYWxsIHR5cGVzDQphZ2Vfc2V4ICU+JSANCiAgZ3JvdXBfYnkocXVhcnRlcikgJT4lIA0KICBzdW1tYXJpc2UoYXZnX3N0YXkgPSBtZWFuKGF2ZXJhZ2VfbGVuZ3RoX29mX3N0YXksIG5hLnJtID0gVFJVRSkpICU+JSANCiAgZ2dwbG90KGFlcyh4ID0gcXVhcnRlciwgeSA9IGF2Z19zdGF5LCBncm91cCA9IDEpKSsNCiAgZ2VvbV9saW5lKGFlcyhncm91cCA9IDEpKSsNCiAgZ2VvbV9wb2ludCgpKw0KICB0aGVtZShheGlzLnRleHQueCA9IGVsZW1lbnRfdGV4dChhbmdsZSA9IDQ1LCBoanVzdCA9IDEpKQ0KDQojIG51bWJlciBvZiBzdGF5cyBmb3IgYWxsIHR5cGVzDQphZ2Vfc2V4ICU+JSANCiAgZ3JvdXBfYnkocXVhcnRlcikgJT4lIA0KICBzdW1tYXJpc2UodG90YWxfc3RheXMgPSBzdW0oc3RheXMsIG5hLnJtID0gVFJVRSkpICU+JSANCiAgZ2dwbG90KGFlcyh4ID0gcXVhcnRlciwgeSA9IHRvdGFsX3N0YXlzLCBncm91cCA9IDEpKSsNCiAgZ2VvbV9saW5lKGFlcyhncm91cCA9IDEpKSsNCiAgZ2VvbV9wb2ludCgpKw0KICB0aGVtZShheGlzLnRleHQueCA9IGVsZW1lbnRfdGV4dChhbmdsZSA9IDQ1LCBoanVzdCA9IDEpKQ0KYGBgDQoNCg0KDQoNCg0KDQoNCg0KDQoNCg0KYGBge3J9DQpzaW1kIDwtIHJlYWRfY3N2KCJyYXdfZGF0YS9ub25fY292aWRfcmF3X2RhdGEvaW5wYXRpZW50X2FuZF9kYXljYXNlX2J5X25oc19ib2FyZF9vZl90cmVhdG1lbnRfYW5kX3NpbWQuY3N2IikgJT4lIGphbml0b3I6OmNsZWFuX25hbWVzKCkNCmBgYA0KYGBge3J9DQojdG90YWwgZXBpc29kZXMoaG9zcGl0YWxpc2F0aW9ucz8pIGJ5IHNpbWQgdmFsdWUNCnNpbWQgJT4lIA0KICBkcm9wX25hKHNpbWQpICU+JQ0KICBtdXRhdGUoc2ltZCA9IGFzLmZhY3RvcihzaW1kKSkgJT4lICMgZ2l2ZXMgZWFjaCBzaW1kIGEgc2VwYXJhdGUgY29sb3VyDQogIGdyb3VwX2J5KHF1YXJ0ZXIsIHNpbWQpICU+JSANCiAgc3VtbWFyaXNlKHRvdGFsX2VwaXNvZGVzID0gc3VtKGVwaXNvZGVzLCBuYS5ybSA9IFRSVUUpKSAlPiUgDQogIGdncGxvdChhZXMoeCA9IHF1YXJ0ZXIsIHkgPSB0b3RhbF9lcGlzb2RlcywgZ3JvdXAgPSBzaW1kKSkrDQogIGdlb21fbGluZShhZXMoY29sb3VyID0gc2ltZCkpKw0KICB0aGVtZShheGlzLnRleHQueCA9IGVsZW1lbnRfdGV4dChhbmdsZSA9IDQ1LCBoanVzdCA9IDEpKSsNCiAgc2NhbGVfeV9jb250aW51b3VzKGxhYmVscyA9IHNjYWxlczo6Y29tbWEpDQogIA0KI2F2ZyBudW1iZXIgb2YgZXBpc29kZXMoaG9zcGl0YWxpc2F0aW9ucz8pIGJ5IHNpbWQgdmFsdWUNCnNpbWQgJT4lIA0KICBkcm9wX25hKHNpbWQpICU+JQ0KICBtdXRhdGUoc2ltZCA9IGFzLmZhY3RvcihzaW1kKSkgJT4lICMgZ2l2ZXMgZWFjaCBzaW1kIGEgc2VwYXJhdGUgY29sb3VyDQogIGdyb3VwX2J5KHF1YXJ0ZXIsIHNpbWQpICU+JSANCiAgc3VtbWFyaXNlKGF2Z19lcGlzb2RlcyA9IG1lYW4oZXBpc29kZXMsIG5hLnJtID0gVFJVRSkpICU+JSANCiAgZ2dwbG90KGFlcyh4ID0gcXVhcnRlciwgeSA9IGF2Z19lcGlzb2RlcywgZ3JvdXAgPSBzaW1kKSkrDQogIGdlb21fbGluZShhZXMoY29sb3VyID0gc2ltZCkpKw0KICB0aGVtZShheGlzLnRleHQueCA9IGVsZW1lbnRfdGV4dChhbmdsZSA9IDQ1LCBoanVzdCA9IDEpKSsNCiAgc2NhbGVfeV9jb250aW51b3VzKGxhYmVscyA9IHNjYWxlczo6Y29tbWEpDQoNCiNhdmcgc3RheXMgYnkgc2ltZCB2YWx1ZQ0Kc2ltZCAlPiUgDQogIGRyb3BfbmEoc2ltZCkgJT4lDQogIG11dGF0ZShzaW1kID0gYXMuZmFjdG9yKHNpbWQpKSAlPiUgIyBnaXZlcyBlYWNoIHNpbWQgYSBzZXBhcmF0ZSBjb2xvdXINCiAgZ3JvdXBfYnkocXVhcnRlciwgc2ltZCkgJT4lIA0KICBzdW1tYXJpc2UoYXZnX3N0YXlzID0gbWVhbihzdGF5cywgbmEucm0gPSBUUlVFKSkgJT4lIA0KICBnZ3Bsb3QoYWVzKHggPSBxdWFydGVyLCB5ID0gYXZnX3N0YXlzLCBncm91cCA9IHNpbWQpKSsNCiAgZ2VvbV9saW5lKGFlcyhjb2xvdXIgPSBzaW1kKSkrDQogIHRoZW1lKGF4aXMudGV4dC54ID0gZWxlbWVudF90ZXh0KGFuZ2xlID0gNDUsIGhqdXN0ID0gMSkpKw0KICBzY2FsZV95X2NvbnRpbnVvdXMobGFiZWxzID0gc2NhbGVzOjpjb21tYSkNCiAgDQpgYGANCmBgYHtyfQ0KI3RvdGFsIGVwaXNvZGVzKGhvc3BpdGFsaXNhdGlvbnM/KSBieSBzaW1kIHZhbHVlDQpzaW1kICU+JSANCiAgZHJvcF9uYShzaW1kKSAlPiUNCiAgZmlsdGVyKGFkbWlzc2lvbl90eXBlID09ICJFbWVyZ2VuY3kgSW5wYXRpZW50cyIpICU+JSANCiAgbXV0YXRlKHNpbWQgPSBhcy5mYWN0b3Ioc2ltZCkpICU+JSAjIGdpdmVzIGVhY2ggc2ltZCBhIHNlcGFyYXRlIGNvbG91cg0KICBncm91cF9ieShxdWFydGVyLCBzaW1kKSAlPiUgDQogIHN1bW1hcmlzZSh0b3RhbF9lcGlzb2RlcyA9IHN1bShlcGlzb2RlcywgbmEucm0gPSBUUlVFKSkgJT4lIA0KICBnZ3Bsb3QoYWVzKHggPSBxdWFydGVyLCB5ID0gdG90YWxfZXBpc29kZXMsIGdyb3VwID0gc2ltZCkpKw0KICBnZW9tX2xpbmUoYWVzKGNvbG91ciA9IHNpbWQpKSsNCiAgdGhlbWUoYXhpcy50ZXh0LnggPSBlbGVtZW50X3RleHQoYW5nbGUgPSA0NSwgaGp1c3QgPSAxKSkrDQogIHNjYWxlX3lfY29udGludW91cyhsYWJlbHMgPSBzY2FsZXM6OmNvbW1hKQ0KICANCiNhdmcgbnVtYmVyIG9mIGVwaXNvZGVzKGhvc3BpdGFsaXNhdGlvbnM/KSBieSBzaW1kIHZhbHVlDQpzaW1kICU+JSANCiAgZHJvcF9uYShzaW1kKSAlPiUNCiAgZmlsdGVyKGFkbWlzc2lvbl90eXBlID09ICJFbWVyZ2VuY3kgSW5wYXRpZW50cyIpICU+JSANCiAgbXV0YXRlKHNpbWQgPSBhcy5mYWN0b3Ioc2ltZCkpICU+JSAjIGdpdmVzIGVhY2ggc2ltZCBhIHNlcGFyYXRlIGNvbG91cg0KICBncm91cF9ieShxdWFydGVyLCBzaW1kKSAlPiUgDQogIHN1bW1hcmlzZShhdmdfZXBpc29kZXMgPSBtZWFuKGVwaXNvZGVzLCBuYS5ybSA9IFRSVUUpKSAlPiUgDQogIGdncGxvdChhZXMoeCA9IHF1YXJ0ZXIsIHkgPSBhdmdfZXBpc29kZXMsIGdyb3VwID0gc2ltZCkpKw0KICBnZW9tX2xpbmUoYWVzKGNvbG91ciA9IHNpbWQpKSsNCiAgdGhlbWUoYXhpcy50ZXh0LnggPSBlbGVtZW50X3RleHQoYW5nbGUgPSA0NSwgaGp1c3QgPSAxKSkrDQogIHNjYWxlX3lfY29udGludW91cyhsYWJlbHMgPSBzY2FsZXM6OmNvbW1hKQ0KDQojYXZnIHN0YXlzIGJ5IHNpbWQgdmFsdWUNCnNpbWQgJT4lIA0KICBkcm9wX25hKHNpbWQpICU+JQ0KICBmaWx0ZXIoYWRtaXNzaW9uX3R5cGUgPT0gIkVtZXJnZW5jeSBJbnBhdGllbnRzIikgJT4lIA0KICBtdXRhdGUoc2ltZCA9IGFzLmZhY3RvcihzaW1kKSkgJT4lICMgZ2l2ZXMgZWFjaCBzaW1kIGEgc2VwYXJhdGUgY29sb3VyDQogIGdyb3VwX2J5KHF1YXJ0ZXIsIHNpbWQpICU+JSANCiAgc3VtbWFyaXNlKGF2Z19zdGF5cyA9IG1lYW4oc3RheXMsIG5hLnJtID0gVFJVRSkpICU+JSANCiAgZ2dwbG90KGFlcyh4ID0gcXVhcnRlciwgeSA9IGF2Z19zdGF5cywgZ3JvdXAgPSBzaW1kKSkrDQogIGdlb21fbGluZShhZXMoY29sb3VyID0gc2ltZCkpKw0KICB0aGVtZShheGlzLnRleHQueCA9IGVsZW1lbnRfdGV4dChhbmdsZSA9IDQ1LCBoanVzdCA9IDEpKSsNCiAgc2NhbGVfeV9jb250aW51b3VzKGxhYmVscyA9IHNjYWxlczo6Y29tbWEpDQogIA0KDQpgYGANCg0KDQpgYGB7cn0NCiMgcGxvdCBhdmcgc3RheSBsZW5ndGggZm9yIG1vc3QgYW5kIGxlYXN0IGRlcHJpdmVkIGZvciBlbWVyZ2VuY3kgdW5wYXRpZW50cw0Kc2ltZCAlPiUgDQogIGZpbHRlcihhZG1pc3Npb25fdHlwZSA9PSAiRW1lcmdlbmN5IElucGF0aWVudHMiLCBzaW1kICVpbiUgYygxLDUpKSAlPiUgDQogIGRyb3BfbmEoc2ltZCkgJT4lIA0KICBncm91cF9ieShxdWFydGVyLHNpbWQpICU+JSANCiAgc3VtbWFyaXNlKGF2Z19zdGF5X2xlbmd0aCA9IG1lYW4oYXZlcmFnZV9sZW5ndGhfb2Zfc3RheSwgbmEucm0gPSBUUlVFKSkgJT4lIA0KICBnZ3Bsb3QoYWVzKHggPSBxdWFydGVyLCB5ID0gYXZnX3N0YXlfbGVuZ3RoKSkgKw0KICBnZW9tX2xpbmUoYWVzKGNvbG91ciA9IHNpbWQsIGdyb3VwID0gc2ltZCkpKw0KICB0aGVtZShheGlzLnRleHQueCA9IGVsZW1lbnRfdGV4dChhbmdsZSA9IDQ1LCBoanVzdCA9IDEpKQ0KICANCmBgYA0KDQoNCmBgYHtyfQ0Kc3BlY2lhbGl0eSA8LSByZWFkX2NzdigicmF3X2RhdGEvbm9uX2NvdmlkX3Jhd19kYXRhL2lucGF0aWVudF9hbmRfZGF5Y2FzZV9ieV9uaHNfYm9hcmRfb2ZfdHJlYXRtZW50X2FuZF9zcGVjaWFsdHkuY3N2IikgJT4lIGphbml0b3I6OmNsZWFuX25hbWVzKCkNCmBgYA0KDQpgYGB7cn0NCnNwZWNpYWxpdHkgJT4lIA0KICBjb3VudChhZG1pc3Npb25fdHlwZSkNCg0Kc3BlY2lhbGl0eSAlPiUgDQogIGNvdW50KGhiKQ0KDQpzcGVjaWFsaXR5ICU+JSANCiAgY291bnQobG9jYXRpb24pDQoNCnNwZWNpYWxpdHkgJT4lIA0KICBjb3VudChzcGVjaWFsdHlfbmFtZSkNCg0KIyBhZGQgYXZlcmFnZXMgDQpzcGVjaWFsaXR5X2F2ZXJhZ2VzIDwtIHNwZWNpYWxpdHkgJT4lIA0KICBncm91cF9ieShxdWFydGVyLCBhZG1pc3Npb25fdHlwZSkgJT4lIA0KICBtdXRhdGUoYXZnX2xlbmd0aF9zcGVsbD0gbWVhbihhdmVyYWdlX2xlbmd0aF9vZl9zcGVsbCwgbmEucm0gPSBUUlVFKSwNCiAgICAgICAgIGF2Z19sZW5ndGhfZXBpc29kZSA9IG1lYW4oYXZlcmFnZV9sZW5ndGhfb2ZfZXBpc29kZSwgbmEucm0gPSBUUlVFKSkgJT4lIA0KICB1bmdyb3VwKCkNCg0KDQpzcGVjaWFsaXR5X2F2ZXJhZ2VzICU+JSANCiAgZ3JvdXBfYnkocXVhcnRlciwgYWRtaXNzaW9uX3R5cGUpICU+JSANCiAgc2xpY2UoMSkgJT4lIA0KICBnZ3Bsb3QoYWVzKHggPSBxdWFydGVyLCB5ID0gYXZlcmFnZV9sZW5ndGhfb2ZfZXBpc29kZSwgZ3JvdXAgPSBhZG1pc3Npb25fdHlwZSkpICsgDQogIGdlb21fbGluZShhZXMoY29sb3VyID0gYWRtaXNzaW9uX3R5cGUpKSsNCiAgdGhlbWUoYXhpcy50ZXh0LnggPSBlbGVtZW50X3RleHQoYW5nbGUgPSA0NSwgaGp1c3QgPSAxKSkNCg0KYGBgDQoNCmBgYHtyfQ0KYWVfd2FpdF90aW1lcyA8LSByZWFkX2NzdigicmF3X2RhdGEvbm9uX2NvdmlkX3Jhd19kYXRhL21vbnRobHlfYWVfd2FpdGluZ3RpbWVzXzIwMjIwNi5jc3YiKSAlPiUgamFuaXRvcjo6Y2xlYW5fbmFtZXMoKQ0KDQojZ2xpbXBzZShhZV93YWl0X3RpbWVzKQ0KDQoNCiNtYWtlIGEgZGF0ZSBhbmQgeWVhciBjb2x1bW4gd2l0aCB0aGUgZmlyc3QgZGF0ZSBvZiBldmVyeSBtb250aA0KYWVfd2FpdF90aW1lcyA8LSBhZV93YWl0X3RpbWVzICU+JSANCiAgbXV0YXRlKGRhdGUgPSB5bShtb250aCksIC5hZnRlciA9IG1vbnRoLA0KICAgICAgICAgeWVhciA9IHllYXIoZGF0ZSkpDQoNCiNtYWtlIGEgcGVyY2VudCBjb2x1bW4gd2l0aCBwZXJjZW50IG9mIHBhdGllbnRzIG1lZXRpbmcgdGhlIDRociB0YXJnZXQgdGltZQ0KYWVfd2FpdF90aW1lcyA8LSBhZV93YWl0X3RpbWVzICU+JSANCiAgbXV0YXRlKHBlcmNlbnRfNGhyX3RhcmdldF9hY2hpZXZlZCA9IChudW1iZXJfbWVldGluZ190YXJnZXRfYWdncmVnYXRlL251bWJlcl9vZl9hdHRlbmRhbmNlc19hZ2dyZWdhdGUpKjEwMCkgJT4lIA0KICAjYWRkIGFuIDhociBvbmUgLSBub3QgY3VycmVudGx5IHVzZWQNCm11dGF0ZShwZXJjZW50X3NlZW5fd2l0aGluXzhociA9ICgobnVtYmVyX29mX2F0dGVuZGFuY2VzX2FnZ3JlZ2F0ZS1hdHRlbmRhbmNlX2dyZWF0ZXI4aHJzKS9udW1iZXJfb2ZfYXR0ZW5kYW5jZXNfYWdncmVnYXRlKSoxMDApDQpgYGANCg0KDQpgYGB7cn0NCiMgZHJhdyBwZXJjZW50YWdlIG9mIDQgaG91ciB3YWl0IGZvciBhbGwgeWVhcnMNCmZvcl9wbG90bHkgPC0gYWVfd2FpdF90aW1lcyAlPiUgDQogIGZpbHRlcihkZXBhcnRtZW50X3R5cGUgPT0gIkVtZXJnZW5jeSBEZXBhcnRtZW50IikgJT4lIA0KICBncm91cF9ieShkYXRlLCBkZXBhcnRtZW50X3R5cGUpICU+JSANCiAgc3VtbWFyaXNlKGF2Z180aHJfdGFyZ2V0X21hZGUgPSBtZWFuKHBlcmNlbnRfNGhyX3RhcmdldF9hY2hpZXZlZCkpICU+JSANCiAgZ2dwbG90KGFlcyh4ID0gZGF0ZSwgeSA9IGF2Z180aHJfdGFyZ2V0X21hZGUpKSsNCiAgZ2VvbV9saW5lKGFlcyhjb2xvdXIgPSBkZXBhcnRtZW50X3R5cGUpKSsNCiAgc2NhbGVfeF9kYXRlKGRhdGVfYnJlYWtzID0gIjYgbW9udGhzIiwgZGF0ZV9sYWJlbHMgPSAgIiViICVZIikrDQogIHRoZW1lKGF4aXMudGV4dC54ID0gZWxlbWVudF90ZXh0KGFuZ2xlID0gOTAsIGhqdXN0ID0gMSwgc2l6ZSA9NykpKw0KICBnZW9tX3Ntb290aCgpKw0KICBnZW9tX3ZsaW5lKHhpbnRlcmNlcHQgPSBhcy5udW1lcmljKGFzLkRhdGUoIjIwMDgtMDEtMDEiKSksIGxpbmV0eXBlPTQpKw0KICBnZW9tX3ZsaW5lKHhpbnRlcmNlcHQgPSBhcy5udW1lcmljKGFzLkRhdGUoIjIwMDktMDEtMDEiKSksIGxpbmV0eXBlPTQpKw0KICBnZW9tX3ZsaW5lKHhpbnRlcmNlcHQgPSBhcy5udW1lcmljKGFzLkRhdGUoIjIwMTAtMDEtMDEiKSksIGxpbmV0eXBlPTQpKw0KICBnZW9tX3ZsaW5lKHhpbnRlcmNlcHQgPSBhcy5udW1lcmljKGFzLkRhdGUoIjIwMTEtMDEtMDEiKSksIGxpbmV0eXBlPTQpKw0KICBnZW9tX3ZsaW5lKHhpbnRlcmNlcHQgPSBhcy5udW1lcmljKGFzLkRhdGUoIjIwMTItMDEtMDEiKSksIGxpbmV0eXBlPTQpKw0KICBnZW9tX3ZsaW5lKHhpbnRlcmNlcHQgPSBhcy5udW1lcmljKGFzLkRhdGUoIjIwMTMtMDEtMDEiKSksIGxpbmV0eXBlPTQpKw0KICBnZW9tX3ZsaW5lKHhpbnRlcmNlcHQgPSBhcy5udW1lcmljKGFzLkRhdGUoIjIwMTQtMDEtMDEiKSksIGxpbmV0eXBlPTQpKw0KICBnZW9tX3ZsaW5lKHhpbnRlcmNlcHQgPSBhcy5udW1lcmljKGFzLkRhdGUoIjIwMTUtMDEtMDEiKSksIGxpbmV0eXBlPTQpKw0KICBnZW9tX3ZsaW5lKHhpbnRlcmNlcHQgPSBhcy5udW1lcmljKGFzLkRhdGUoIjIwMTYtMDEtMDEiKSksIGxpbmV0eXBlPTQpKw0KICBnZW9tX3ZsaW5lKHhpbnRlcmNlcHQgPSBhcy5udW1lcmljKGFzLkRhdGUoIjIwMTctMDEtMDEiKSksIGxpbmV0eXBlPTQpKw0KICBnZW9tX3ZsaW5lKHhpbnRlcmNlcHQgPSBhcy5udW1lcmljKGFzLkRhdGUoIjIwMTgtMDEtMDEiKSksIGxpbmV0eXBlPTQpKw0KICBnZW9tX3ZsaW5lKHhpbnRlcmNlcHQgPSBhcy5udW1lcmljKGFzLkRhdGUoIjIwMTktMDEtMDEiKSksIGxpbmV0eXBlPTQpKw0KICBnZW9tX3ZsaW5lKHhpbnRlcmNlcHQgPSBhcy5udW1lcmljKGFzLkRhdGUoIjIwMjAtMDEtMDEiKSksIGxpbmV0eXBlPTQpKw0KICBsYWJzKHRpdGxlID0gInBlcmNlbnRhZ2Ugb2YgQSZFIGRlcGFydG1lbnRzIG1lZXRpbmcgdGhlIDQgaHIgdGFyZ2V0IHR1cm5hcm91bmQgZm9yIHBhdGllbnRzIiwNCiAgICAgICBzdWJ0aXRsZSA9ICJhZGRlZCBpbiB2ZXJ0aWNhbCBsaW5lcyBmb3IgSmFudWFyeSB0byBoZWxwIikNCg0KZ2dwbG90bHkoZm9yX3Bsb3RseSkNCmBgYA0KDQpgYGB7cn0NCiMgNGhyIHdhaXQgYnkgaGVhbHRoIGJvYXJkIGZvciBhbGwgeWVhcnMgZmFjZXQgd3JhcHBlZA0KYWVfd2FpdF90aW1lcyAlPiUgDQogIGZpbHRlcihkZXBhcnRtZW50X3R5cGUgPT0gIkVtZXJnZW5jeSBEZXBhcnRtZW50IikgJT4lIA0KICBncm91cF9ieShkYXRlLCBoYnQpICU+JSANCiAgbXV0YXRlKGF2Z180aHJfdGFyZ2V0X21hZGUgPSBtZWFuKHBlcmNlbnRfNGhyX3RhcmdldF9hY2hpZXZlZCkpICU+JSANCiAgc2xpY2UoMSkgJT4lICANCiAgZ2dwbG90KGFlcyh4ID0gZGF0ZSwgeSA9IGF2Z180aHJfdGFyZ2V0X21hZGUpKSsNCiAgZ2VvbV9saW5lKGFlcyhjb2xvdXIgPSBoYnQpKSsNCiAgc2NhbGVfeF9kYXRlKGRhdGVfYnJlYWtzID0gIjYgbW9udGhzIiwgZGF0ZV9sYWJlbHMgPSAgIiViICVZIikrDQogIHRoZW1lKGF4aXMudGV4dC54ID0gZWxlbWVudF90ZXh0KGFuZ2xlID0gOTAsIGhqdXN0ID0gMSwgc2l6ZSA9NykpKw0KICBnZW9tX3Ntb290aCgpKw0KICBnZW9tX3ZsaW5lKHhpbnRlcmNlcHQgPSBhcy5udW1lcmljKGFzLkRhdGUoIjIwMDgtMDEtMDEiKSksIGxpbmV0eXBlPTQpKw0KICBnZW9tX3ZsaW5lKHhpbnRlcmNlcHQgPSBhcy5udW1lcmljKGFzLkRhdGUoIjIwMDktMDEtMDEiKSksIGxpbmV0eXBlPTQpKw0KICBnZW9tX3ZsaW5lKHhpbnRlcmNlcHQgPSBhcy5udW1lcmljKGFzLkRhdGUoIjIwMTAtMDEtMDEiKSksIGxpbmV0eXBlPTQpKw0KICBnZW9tX3ZsaW5lKHhpbnRlcmNlcHQgPSBhcy5udW1lcmljKGFzLkRhdGUoIjIwMTEtMDEtMDEiKSksIGxpbmV0eXBlPTQpKw0KICBnZW9tX3ZsaW5lKHhpbnRlcmNlcHQgPSBhcy5udW1lcmljKGFzLkRhdGUoIjIwMTItMDEtMDEiKSksIGxpbmV0eXBlPTQpKw0KICBnZW9tX3ZsaW5lKHhpbnRlcmNlcHQgPSBhcy5udW1lcmljKGFzLkRhdGUoIjIwMTMtMDEtMDEiKSksIGxpbmV0eXBlPTQpKw0KICBnZW9tX3ZsaW5lKHhpbnRlcmNlcHQgPSBhcy5udW1lcmljKGFzLkRhdGUoIjIwMTQtMDEtMDEiKSksIGxpbmV0eXBlPTQpKw0KICBnZW9tX3ZsaW5lKHhpbnRlcmNlcHQgPSBhcy5udW1lcmljKGFzLkRhdGUoIjIwMTUtMDEtMDEiKSksIGxpbmV0eXBlPTQpKw0KICBnZW9tX3ZsaW5lKHhpbnRlcmNlcHQgPSBhcy5udW1lcmljKGFzLkRhdGUoIjIwMTYtMDEtMDEiKSksIGxpbmV0eXBlPTQpKw0KICBnZW9tX3ZsaW5lKHhpbnRlcmNlcHQgPSBhcy5udW1lcmljKGFzLkRhdGUoIjIwMTctMDEtMDEiKSksIGxpbmV0eXBlPTQpKw0KICBnZW9tX3ZsaW5lKHhpbnRlcmNlcHQgPSBhcy5udW1lcmljKGFzLkRhdGUoIjIwMTgtMDEtMDEiKSksIGxpbmV0eXBlPTQpKw0KICBnZW9tX3ZsaW5lKHhpbnRlcmNlcHQgPSBhcy5udW1lcmljKGFzLkRhdGUoIjIwMTktMDEtMDEiKSksIGxpbmV0eXBlPTQpKw0KICBmYWNldF93cmFwKH4gaGJ0KQ0KYGBgDQoNCmBgYHtyfQ0KdGFyZ2V0XzIwMDcgPC0gYWVfd2FpdF90aW1lcyAlPiUNCiAgZ3JvdXBfYnkoeWVhciwgaGJ0KSAlPiUgDQogIHN1bW1hcmlzZShhZV80aHJfdGFyZ2V0X2FjaGlldmVkID0gbWVhbihwZXJjZW50XzRocl90YXJnZXRfYWNoaWV2ZWQsIG5hLnJtID0gVFJVRSkpICU+JSANCiAgZmlsdGVyKHllYXIgPT0gMjAwNykgJT4lIA0KICByZW5hbWUoYWVfdGFyZ2V0XzIwMDcgPSBhZV80aHJfdGFyZ2V0X2FjaGlldmVkKSAlPiUgDQogIHVuZ3JvdXAoKSAlPiUgDQogIHNlbGVjdChoYnQsYWVfdGFyZ2V0XzIwMDcpDQoNCnRhcmdldF8yMDA4IDwtIGFlX3dhaXRfdGltZXMgJT4lDQogIGdyb3VwX2J5KHllYXIsIGhidCkgJT4lIA0KICBzdW1tYXJpc2UoYWVfNGhyX3RhcmdldF9hY2hpZXZlZCA9IG1lYW4ocGVyY2VudF80aHJfdGFyZ2V0X2FjaGlldmVkLCBuYS5ybSA9IFRSVUUpKSAlPiUgDQogIGZpbHRlcih5ZWFyID09IDIwMDgpICU+JSANCiAgcmVuYW1lKGFlX3RhcmdldF8yMDA4ID0gYWVfNGhyX3RhcmdldF9hY2hpZXZlZCkgJT4lIA0KICB1bmdyb3VwKCkgJT4lIA0KICBzZWxlY3QoaGJ0LGFlX3RhcmdldF8yMDA4KQ0KDQp0YXJnZXRfMjAwOSA8LSBhZV93YWl0X3RpbWVzICU+JQ0KICBncm91cF9ieSh5ZWFyLCBoYnQpICU+JSANCiAgc3VtbWFyaXNlKGFlXzRocl90YXJnZXRfYWNoaWV2ZWQgPSBtZWFuKHBlcmNlbnRfNGhyX3RhcmdldF9hY2hpZXZlZCwgbmEucm0gPSBUUlVFKSkgJT4lIA0KICBmaWx0ZXIoeWVhciA9PSAyMDA5KSAlPiUgDQogIHJlbmFtZShhZV90YXJnZXRfMjAwOSA9IGFlXzRocl90YXJnZXRfYWNoaWV2ZWQpICU+JSANCiAgdW5ncm91cCgpICU+JSANCiAgc2VsZWN0KGhidCxhZV90YXJnZXRfMjAwOSkNCg0KdGFyZ2V0XzIwMTAgPC0gYWVfd2FpdF90aW1lcyAlPiUNCiAgZ3JvdXBfYnkoeWVhciwgaGJ0KSAlPiUgDQogIHN1bW1hcmlzZShhZV80aHJfdGFyZ2V0X2FjaGlldmVkID0gbWVhbihwZXJjZW50XzRocl90YXJnZXRfYWNoaWV2ZWQsIG5hLnJtID0gVFJVRSkpICU+JSANCiAgZmlsdGVyKHllYXIgPT0gMjAxMCkgJT4lIA0KICByZW5hbWUoYWVfdGFyZ2V0XzIwMTAgPSBhZV80aHJfdGFyZ2V0X2FjaGlldmVkKSAlPiUgDQogIHVuZ3JvdXAoKSAlPiUgDQogIHNlbGVjdChoYnQsYWVfdGFyZ2V0XzIwMTApDQoNCnRhcmdldF8yMDExIDwtIGFlX3dhaXRfdGltZXMgJT4lDQogIGdyb3VwX2J5KHllYXIsIGhidCkgJT4lIA0KICBzdW1tYXJpc2UoYWVfNGhyX3RhcmdldF9hY2hpZXZlZCA9IG1lYW4ocGVyY2VudF80aHJfdGFyZ2V0X2FjaGlldmVkLCBuYS5ybSA9IFRSVUUpKSAlPiUgDQogIGZpbHRlcih5ZWFyID09IDIwMTEpICU+JSANCiAgcmVuYW1lKGFlX3RhcmdldF8yMDExID0gYWVfNGhyX3RhcmdldF9hY2hpZXZlZCkgJT4lIA0KICB1bmdyb3VwKCkgJT4lIA0KICBzZWxlY3QoaGJ0LGFlX3RhcmdldF8yMDExKQ0KDQp0YXJnZXRfMjAxMiA8LSBhZV93YWl0X3RpbWVzICU+JQ0KICBncm91cF9ieSh5ZWFyLCBoYnQpICU+JSANCiAgc3VtbWFyaXNlKGFlXzRocl90YXJnZXRfYWNoaWV2ZWQgPSBtZWFuKHBlcmNlbnRfNGhyX3RhcmdldF9hY2hpZXZlZCwgbmEucm0gPSBUUlVFKSkgJT4lIA0KICBmaWx0ZXIoeWVhciA9PSAyMDEyKSAlPiUgDQogIHJlbmFtZShhZV90YXJnZXRfMjAxMiA9IGFlXzRocl90YXJnZXRfYWNoaWV2ZWQpICU+JSANCiAgdW5ncm91cCgpICU+JSANCiAgc2VsZWN0KGhidCxhZV90YXJnZXRfMjAxMikNCg0KdGFyZ2V0XzIwMTMgPC0gYWVfd2FpdF90aW1lcyAlPiUNCiAgZ3JvdXBfYnkoeWVhciwgaGJ0KSAlPiUgDQogIHN1bW1hcmlzZShhZV80aHJfdGFyZ2V0X2FjaGlldmVkID0gbWVhbihwZXJjZW50XzRocl90YXJnZXRfYWNoaWV2ZWQsIG5hLnJtID0gVFJVRSkpICU+JSANCiAgZmlsdGVyKHllYXIgPT0gMjAxMykgJT4lIA0KICByZW5hbWUoYWVfdGFyZ2V0XzIwMTMgPSBhZV80aHJfdGFyZ2V0X2FjaGlldmVkKSAlPiUgDQogIHVuZ3JvdXAoKSAlPiUgDQogIHNlbGVjdChoYnQsYWVfdGFyZ2V0XzIwMTMpDQoNCnRhcmdldF8yMDE0IDwtIGFlX3dhaXRfdGltZXMgJT4lDQogIGdyb3VwX2J5KHllYXIsIGhidCkgJT4lIA0KICBzdW1tYXJpc2UoYWVfNGhyX3RhcmdldF9hY2hpZXZlZCA9IG1lYW4ocGVyY2VudF80aHJfdGFyZ2V0X2FjaGlldmVkLCBuYS5ybSA9IFRSVUUpKSAlPiUgDQogIGZpbHRlcih5ZWFyID09IDIwMTQpICU+JSANCiAgcmVuYW1lKGFlX3RhcmdldF8yMDE0ID0gYWVfNGhyX3RhcmdldF9hY2hpZXZlZCkgJT4lIA0KICB1bmdyb3VwKCkgJT4lIA0KICBzZWxlY3QoaGJ0LGFlX3RhcmdldF8yMDE0KQ0KDQp0YXJnZXRfMjAxNSA8LSBhZV93YWl0X3RpbWVzICU+JQ0KICBncm91cF9ieSh5ZWFyLCBoYnQpICU+JSANCiAgc3VtbWFyaXNlKGFlXzRocl90YXJnZXRfYWNoaWV2ZWQgPSBtZWFuKHBlcmNlbnRfNGhyX3RhcmdldF9hY2hpZXZlZCwgbmEucm0gPSBUUlVFKSkgJT4lIA0KICBmaWx0ZXIoeWVhciA9PSAyMDE1KSAlPiUgDQogIHJlbmFtZShhZV90YXJnZXRfMjAxNSA9IGFlXzRocl90YXJnZXRfYWNoaWV2ZWQpICU+JSANCiAgdW5ncm91cCgpICU+JSANCiAgc2VsZWN0KGhidCxhZV90YXJnZXRfMjAxNSkNCg0KdGFyZ2V0XzIwMTYgPC0gYWVfd2FpdF90aW1lcyAlPiUNCiAgZ3JvdXBfYnkoeWVhciwgaGJ0KSAlPiUgDQogIHN1bW1hcmlzZShhZV80aHJfdGFyZ2V0X2FjaGlldmVkID0gbWVhbihwZXJjZW50XzRocl90YXJnZXRfYWNoaWV2ZWQsIG5hLnJtID0gVFJVRSkpICU+JSANCiAgZmlsdGVyKHllYXIgPT0gMjAxNikgJT4lIA0KICByZW5hbWUoYWVfdGFyZ2V0XzIwMTYgPSBhZV80aHJfdGFyZ2V0X2FjaGlldmVkKSAlPiUgDQogIHVuZ3JvdXAoKSAlPiUgDQogIHNlbGVjdChoYnQsYWVfdGFyZ2V0XzIwMTYpDQoNCnRhcmdldF8yMDE3IDwtIGFlX3dhaXRfdGltZXMgJT4lDQogIGdyb3VwX2J5KHllYXIsIGhidCkgJT4lIA0KICBzdW1tYXJpc2UoYWVfNGhyX3RhcmdldF9hY2hpZXZlZCA9IG1lYW4ocGVyY2VudF80aHJfdGFyZ2V0X2FjaGlldmVkLCBuYS5ybSA9IFRSVUUpKSAlPiUgDQogIGZpbHRlcih5ZWFyID09IDIwMTcpICU+JSANCiAgcmVuYW1lKGFlX3RhcmdldF8yMDE3ID0gYWVfNGhyX3RhcmdldF9hY2hpZXZlZCkgJT4lIA0KICB1bmdyb3VwKCkgJT4lIA0KICBzZWxlY3QoaGJ0LGFlX3RhcmdldF8yMDE3KQ0KDQp0YXJnZXRfMjAxOCA8LSBhZV93YWl0X3RpbWVzICU+JQ0KICBncm91cF9ieSh5ZWFyLCBoYnQpICU+JSANCiAgc3VtbWFyaXNlKGFlXzRocl90YXJnZXRfYWNoaWV2ZWQgPSBtZWFuKHBlcmNlbnRfNGhyX3RhcmdldF9hY2hpZXZlZCwgbmEucm0gPSBUUlVFKSkgJT4lIA0KICBmaWx0ZXIoeWVhciA9PSAyMDE4KSAlPiUgDQogIHJlbmFtZShhZV90YXJnZXRfMjAxOCA9IGFlXzRocl90YXJnZXRfYWNoaWV2ZWQpICU+JSANCiAgdW5ncm91cCgpICU+JSANCiAgc2VsZWN0KGhidCxhZV90YXJnZXRfMjAxOCkNCg0KdGFyZ2V0XzIwMTkgPC0gYWVfd2FpdF90aW1lcyAlPiUNCiAgZ3JvdXBfYnkoeWVhciwgaGJ0KSAlPiUgDQogIHN1bW1hcmlzZShhZV80aHJfdGFyZ2V0X2FjaGlldmVkID0gbWVhbihwZXJjZW50XzRocl90YXJnZXRfYWNoaWV2ZWQsIG5hLnJtID0gVFJVRSkpICU+JSANCiAgZmlsdGVyKHllYXIgPT0gMjAxOSkgJT4lIA0KICByZW5hbWUoYWVfdGFyZ2V0XzIwMTkgPSBhZV80aHJfdGFyZ2V0X2FjaGlldmVkKSAlPiUgDQogIHVuZ3JvdXAoKSAlPiUgDQogIHNlbGVjdChoYnQsYWVfdGFyZ2V0XzIwMTkpDQoNCnRhcmdldF8yMDIwIDwtIGFlX3dhaXRfdGltZXMgJT4lDQogIGdyb3VwX2J5KHllYXIsIGhidCkgJT4lIA0KICBzdW1tYXJpc2UoYWVfNGhyX3RhcmdldF9hY2hpZXZlZCA9IG1lYW4ocGVyY2VudF80aHJfdGFyZ2V0X2FjaGlldmVkLCBuYS5ybSA9IFRSVUUpKSAlPiUgDQogIGZpbHRlcih5ZWFyID09IDIwMjApICU+JSANCiAgcmVuYW1lKGFlX3RhcmdldF8yMDIwID0gYWVfNGhyX3RhcmdldF9hY2hpZXZlZCkgJT4lIA0KICB1bmdyb3VwKCkgJT4lIA0KICBzZWxlY3QoaGJ0LGFlX3RhcmdldF8yMDIwKQ0KDQp0YXJnZXRfMjAyMSA8LSBhZV93YWl0X3RpbWVzICU+JQ0KICBncm91cF9ieSh5ZWFyLCBoYnQpICU+JSANCiAgc3VtbWFyaXNlKGFlXzRocl90YXJnZXRfYWNoaWV2ZWQgPSBtZWFuKHBlcmNlbnRfNGhyX3RhcmdldF9hY2hpZXZlZCwgbmEucm0gPSBUUlVFKSkgJT4lIA0KICBmaWx0ZXIoeWVhciA9PSAyMDIxKSAlPiUgDQogIHJlbmFtZShhZV90YXJnZXRfMjAyMSA9IGFlXzRocl90YXJnZXRfYWNoaWV2ZWQpICU+JSANCiAgdW5ncm91cCgpICU+JSANCiAgc2VsZWN0KGhidCxhZV90YXJnZXRfMjAyMSkNCg0KYGBgDQoNCg0KDQoNCg0KDQpgYGB7cn0NCmxpYnJhcnkoc2YpDQoNCnNjb3RsYW5kIDwtIHN0X3JlYWQoIi4uL1NHX05IU19IZWFsdGhCb2FyZHNfMjAxOV9zaGFwZWZpbGUvU0dfTkhTX0hlYWx0aEJvYXJkc18yMDE5LnNocCIpDQoNCiMgdmlldyhzY290bGFuZCkNCiMgDQpoZWFkKHNjb3RsYW5kLCAzKQ0KIyANCnBsb3Qoc2NvdGxhbmRbLTFdKQ0KDQpzY290bGFuZCA8LSAgc2NvdGxhbmQgJT4lIA0KICBtdXRhdGUoY2VudHJlcyA9IHN0X2NlbnRyb2lkKHN0X21ha2VfdmFsaWQoZ2VvbWV0cnkpKSkgJT4lDQogICAgbXV0YXRlKGxhdCA9IHN0X2Nvb3JkaW5hdGVzKGNlbnRyZXMpWywxXSwNCiAgICAgICAgICAgbG9uZyA9IHN0X2Nvb3JkaW5hdGVzKGNlbnRyZXMpWywyXSwNCiAgICAgICAgICAgdGFyZ2V0XzIwMDcgPSB0YXJnZXRfMjAwNyRhZV90YXJnZXRfMjAwNywNCiAgICAgICAgICAgdGFyZ2V0XzIwMDggPSB0YXJnZXRfMjAwOCRhZV90YXJnZXRfMjAwOCwNCiAgICAgICAgICAgdGFyZ2V0XzIwMDkgPSB0YXJnZXRfMjAwOSRhZV90YXJnZXRfMjAwOSwNCiAgICAgICAgICAgdGFyZ2V0XzIwMTAgPSB0YXJnZXRfMjAxMCRhZV90YXJnZXRfMjAxMCwNCiAgICAgICAgICAgdGFyZ2V0XzIwMTEgPSB0YXJnZXRfMjAxMSRhZV90YXJnZXRfMjAxMSwNCiAgICAgICAgICAgdGFyZ2V0XzIwMTIgPSB0YXJnZXRfMjAxMiRhZV90YXJnZXRfMjAxMiwNCiAgICAgICAgICAgdGFyZ2V0XzIwMTMgPSB0YXJnZXRfMjAxMyRhZV90YXJnZXRfMjAxMywNCiAgICAgICAgICAgdGFyZ2V0XzIwMTQgPSB0YXJnZXRfMjAxNCRhZV90YXJnZXRfMjAxNCwNCiAgICAgICAgICAgdGFyZ2V0XzIwMTUgPSB0YXJnZXRfMjAxNSRhZV90YXJnZXRfMjAxNSwNCiAgICAgICAgICAgdGFyZ2V0XzIwMTYgPSB0YXJnZXRfMjAxNiRhZV90YXJnZXRfMjAxNiwNCiAgICAgICAgICAgdGFyZ2V0XzIwMTcgPSB0YXJnZXRfMjAxNyRhZV90YXJnZXRfMjAxNywNCiAgICAgICAgICAgdGFyZ2V0XzIwMTggPSB0YXJnZXRfMjAxOCRhZV90YXJnZXRfMjAxOCwNCiAgICAgICAgICAgdGFyZ2V0XzIwMTkgPSB0YXJnZXRfMjAxOSRhZV90YXJnZXRfMjAxOSwNCiAgICAgICAgICAgdGFyZ2V0XzIwMjAgPSB0YXJnZXRfMjAyMCRhZV90YXJnZXRfMjAyMCwNCiAgICAgICAgICAgdGFyZ2V0XzIwMjEgPSB0YXJnZXRfMjAyMSRhZV90YXJnZXRfMjAyMQ0KICAgICAgICAgICAgICAgICAgKQ0KDQpnZ3Bsb3QoZGF0YSA9IHNjb3RsYW5kKSArDQogICAgZ2VvbV9zZihhZXMoZmlsbCA9IGNoYW5nZV9hZSkpICsNCiAgICBzY2FsZV9maWxsX3ZpcmlkaXNfYyhvcHRpb24gPSAicGxhc21hIikrDQogIGxhYnModGl0bGUgPSAicGVyY2VudCBjaGFuZ2UgaW4gQSZFIGRlcHRzIG1lZXRpbmcgdGhlIDQgaG91ciB0YXJnZXQgMjAwNyAtIDIwMTgiKQ0KICANCg0KZ2dwbG90KGRhdGEgPSBzY290bGFuZCkgKw0KICAgIGdlb21fc2YoYWVzKGZpbGwgPSB0YXJnZXRfMjAxOCkpICsNCiAgICBzY2FsZV9maWxsX3ZpcmlkaXNfYyhvcHRpb24gPSAicGxhc21hIikrDQogIHRoZW1lX3ZvaWQoKSsNCiAgbGFicyh0aXRsZSA9ICJQZXJjZW50IG9mIEEmRSBkZXB0cyBtYWtpbmcgdGhlIDRociB0YXJnZXQiKQ0KYGBgDQoNCg0KYGBge3J9DQpnZ3Bsb3QoZGF0YSA9IHNjb3RsYW5kKSArDQpnZW9tX3NmKGZpbGwgPSAiZ3JlZW4iKSsNCmdncmVwZWw6Omdlb21fdGV4dF9yZXBlbChhZXMoeCA9IGxhdCAsIHkgPSBsb25nLCBsYWJlbCA9IHBhc3RlKEhCQ29kZSwgSEJOYW1lLCBzZXAgPSAiXG4iKSksIG1pbi5zZWdtZW50Lmxlbmd0aCA9IDAuMDUsc2l6ZSA9IDIsIGNvbG9yID0gImJsYWNrIiwgZm9udGZhY2UgPSAiYm9sZCIpICsNCiAgdGhlbWVfdm9pZCgpDQpgYGANCg0KDQpgYGB7cn0NCg0KbGlicmFyeShzZikNCg0Kc2NvdGxhbmRfc21hbGxlciA8LSBzY290bGFuZCAlPiUgIyBtYWtlIGEgc21hbGxlciB2ZXJzaW9uIGZvciBwZXJmb3JtYW5jZSBpc3N1ZXMNCiAgc3Rfc2ltcGxpZnkoVFJVRSwgZFRvbGVyYW5jZSA9IDIwMDApDQojZml4ZXMgcHJvYmxlbXMgY2F1c2VkIGJ5IGFib3ZlIA0Kc2NvdGxhbmRfc21hbGxlciA8LSBzZjo6c3RfY2FzdChzY290bGFuZF9zbWFsbGVyLCAiTVVMVElQT0xZR09OIikNCg0KDQojIA0KIyAgIGZpZyA8LSBnZ3Bsb3RseSgNCiMgICAgIGdncGxvdChzY290bGFuZCkrDQojICAgZ2VvbV9zZihhZXMoZmlsbCA9IEhCTmFtZSkpDQojICkNCiMgICBmaWcNCg0KICANCiAgcCA8LSBnZ3Bsb3Qoc2NvdGxhbmRfc21hbGxlcikgKyANCiAgZ2VvbV9zZihhZXMoZmlsbCA9IEhCTmFtZSwgdGV4dCA9IHBhc3RlKCI8Yj4iLCBIQk5hbWUsICI8L2I+XG4iLCBIQkNvZGUpKSkrDQogICAgdGhlbWVfdm9pZCgpDQogIHAgJT4lDQogIGdncGxvdGx5KHRvb2x0aXAgPSAidGV4dCIpICU+JQ0KICBzdHlsZShob3ZlcmxhYmVsID0gbGlzdChiZ2NvbG9yID0gIndoaXRlIiksIGhvdmVyb24gPSAiZmlsbCIpDQogICAgDQogIA0KYGBgDQoNCi0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tDQoNCg0KYGBge3J9DQpjb3ZpZF9hZ2Vfc2V4IDwtIHJlYWRfY3N2KCJyYXdfZGF0YS9jb3ZpZF9yYXdfZGF0YS9ob3NwaXRhbF9hZG1pc3Npb25zX2hiX2FnZXNleF8yMDIyMDMwMi5jc3YiKQ0KDQpoZWFkKGNvdmlkX2FnZV9zZXgpDQpgYGANCg0KYGBge3J9DQpsaWJyYXJ5KGxlYWZsZXQpDQoNCmRhdGEocXVha2VzKQ0KDQojIFNob3cgZmlyc3QgMjAgcm93cyBmcm9tIHRoZSBgcXVha2VzYCBkYXRhc2V0DQpsZWFmbGV0KGRhdGEgPSBxdWFrZXNbMToyMCxdKSAlPiUgYWRkVGlsZXMoKSAlPiUNCiAgYWRkTWFya2Vycyh+bG9uZywgfmxhdCwgcG9wdXAgPSB+YXMuY2hhcmFjdGVyKG1hZykpDQpgYGANCg0K